3

我有一个应用程序进行大量字符串操作,我注意到 jvm 内存使用率非常高,而我什至没有在内存中存储任何内容。

我制作了一个示例应用程序来演示这个问题:我有一个带有一个按钮的简单 JFrame

private void buttonDoWorkActionPerformed(java.awt.event.ActionEvent evt) {
    String randomString = "abc test garbagecollector";

    ArrayList<String> results;

    for(int i=0; i<100000; i++)
        results = doNothing(randomString.split(" "));

    results = null;//No effect
    System.gc();//No effect
}

private static ArrayList<String> doNothing(String[] words) {
    ArrayList<String> results = new ArrayList<String>();

    for (String word : words)
        results.add(word);

    return results;
}

如果你运行这个示例,JVM 会在内存中占用大约 50Mo,一旦你按下按钮,它就会增加到 150Mo,并且永远不会下降。编辑:我指的是 Windows 任务管理器中的“java.exe”进程。

显然,我正在做很多临时字符串副本,但我希望一旦我丢失了引用,它们会被 carbage 收集器释放。

编辑:也许不相关,但我尝试使用 32 位和 64 位版本的 java 1.6。

4

2 回答 2

4

您需要澄清您的意思,it因为您可能会在这里看到许多级别。有些会在 GC 上减少,有些不会。

如果您在 GC 之后查看内存使用情况,您应该会看到没有增加,如果您看到其他方面的增加,则可能是最大内存或其他一些数字。

public static long memoryUsed() {
    return Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory();
}

public static void main(String... args) {
    String randomString = "abc test garbagecollector";

    System.gc();
    System.out.printf("Memory used before test started %.1f MB%n", memoryUsed() / 1e6);
    for (int t = 0; t < 5; t++) {
        List<String> results = new ArrayList<String>();

        for (int i = 0; i < 100000; i++) {
            for (String word : randomString.split(" "))
                results.add(word);
        }
        long beforeGC = memoryUsed();
        results = null;
        System.gc();
        System.out.printf("%d: before GC %.1f MB used, after GC %.1f MB used%n",
                t, beforeGC / 1e6, memoryUsed() / 1e6);
    }
}

印刷

Memory used before test started 5.7 MB
0: before GC 61.8 MB used, after GC 5.8 MB used
1: before GC 44.3 MB used, after GC 5.8 MB used
2: before GC 44.3 MB used, after GC 5.8 MB used
3: before GC 44.3 MB used, after GC 5.8 MB used
4: before GC 44.3 MB used, after GC 5.8 MB used
于 2012-12-10T16:55:09.490 回答
0

字符串可能正在被实习,这是与一些 VM 设计决策相关的 Java 和字符串问题。您可能可以尝试获取 VisualVM 或其他一些框架,连接到您正在运行的 JVM 并查看对象,和/或进行堆转储并查看结果。

于 2012-12-10T16:54:19.523 回答