加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 编程开发 > Java > 正文

Java的RAM使用率与任务管理器所说的不对应

发布时间:2020-12-15 02:49:12 所属栏目:Java 来源:网络整理
导读:我通过制作1024 ^ 3(基本上是1Gb)长度的字节数组来玩 Java的JVM.我在使用任务管理器(查看进程)和这个小片段之前,在数组创建之后和数组被垃圾收集器销毁之后测量了RAM的使用情况: public static void showMemory() { System.out.println("Memory used: " + (
我通过制作1024 ^ 3(基本上是1Gb)长度的字节数组来玩 Java的JVM.我在使用任务管理器(查看进程)和这个小片段之前,在数组创建之后和数组被垃圾收集器销毁之后测量了RAM的使用情况:
public static void showMemory() {
    System.out.println("Memory used: "
            + (Runtime.getRuntime().totalMemory() - Runtime.getRuntime().freeMemory()) / (1024.D * 1024.D) + "mB.");
}

上述代码分别显示2Mb,1029Mb和2Mb. – >这一切似乎都很正常.
但是,在查看TaskManager时,Java的RAM使用率最初为2mb,然后转到1052Mb并保持不变,即使该代码段显示为2Mb.

由于我希望Java使用最少的资源,我该如何解决这个问题?

编辑:

我做了一些研究并找出了要使用的术语.实际上,本机内存与堆内存的值不相似,并且通常大于堆内存.有没有办法减少使用的本机内存,使其接近堆内存?

解决方法

结论:

使用垃圾优先(G1)GC(Java 9中的默认GC),与ParallelOldGC相比,这个垃圾收集器还缩小了堆大小(最终,它还会缩小garabage集合中使用的整体“本机内存”)( Java 7和Java 8中的默认GC,很少永远不会缩小堆大小!

通常:

你的基本假设是错误的.

您假设代码段显示堆大小.这是不正确的.它显示了堆利用率.这意味着“我的堆使用了多少空间?”. Runtime.getRuntime().totalMemory()显示堆大小Runtime.getRuntime().freeMemory()显示空闲堆大小,它们的差异显示堆利用率(使用的大小).

您的堆以初始大小开始,利用0字节,因为尚未创建任何对象,以及最大堆大小.最大堆大小描述允许垃圾收集器调整堆大小的大小(例如,如果没有足够的空间用于非常大的对象)

在创建空堆之后的下一步,自动加载一些对象(类对象等),它们通常应该很容易适合初始堆大小.

然后,您的代码开始运行并分配对象.只要您的伊甸园空间没有更多空间(堆被分成年轻一代(伊甸园,幸存者和幸存者 – 太空)和老一代,如果您对这些细节感兴趣,请查找其他资源),触发垃圾收集.

在垃圾收集期间,垃圾收集器可能决定调整堆的大小(如上所述,在讨论最大堆大小时).在您的情况下会发生这种情况,因为您的初始堆大小太小而不适合您的1GB对象.因此,堆大小会增加,介于初始堆大小和最大堆大小之间.

然后,在你的大对象死亡之后,下一个GC可以使堆再次变小,但它没有必要.为什么?它低于最大堆大小,这是GC所关心的.一些垃圾收集算法再次缩小堆,有些则没有.

特别是ParallelOldGC,Java 7和Java 8中的默认GC,很少会永远不会缩小堆.

如果您希望GC在垃圾收集期间通过缩小堆来尝试保持较小的堆大小,请通过设置-XX:UseG1GC Java标记来尝试首先使用garabage(G1)GC.

例:

这将以字节打印出所有值.

您将获得概述,两个GC如何工作以及使用其中任何一个时使用的空间.

System.out.println(String.format("Init:t%,d",ManagementFactory.getMemoryMXBean().getHeapMemoryUsage().getInit()));
System.out.println(String.format("Max:t%,d%n",ManagementFactory.getMemoryMXBean().getHeapMemoryUsage().getMax()));

Thread outputThread = new Thread(() -> {
    try {
        int i = 0;
        for(;;) {
            System.out.println(String.format("%dmst->tUsed:tt%,i,ManagementFactory.getMemoryMXBean().getHeapMemoryUsage().getUsed()));
            System.out.println(String.format("%dmst->tCommited:t%,ManagementFactory.getMemoryMXBean().getHeapMemoryUsage().getCommitted()));
            Thread.sleep(100);
            i += 100;
        }
    } catch (Exception e) { }
});

Thread allocThread = new Thread(() -> {
    try {
        int val = 0;
        Thread.sleep(500); // Wait 1/2 second
        createArray();
        Thread.sleep(500); // Wait another 1/2 seconds
        System.gc(); // Force a GC,array should be cleaned
        return;
    } catch (Exception e) { }
});

outputThread.start();
allocThread.start();

createArray()只是以下小方法:

private static void createArray() {
    byte[] arr = new byte[1024 * 1024 * 1024];
}

–Result ParallelOldGC:

Init:   262,144,000
Max:    3,715,629,056

0ms ->  Used:       6,606,272
0ms ->  Commited:   251,658,240
100ms   ->  Used:       6,272
100ms   ->  Commited:   251,240
200ms   ->  Used:       6,272
200ms   ->  Commited:   251,240
300ms   ->  Used:       6,272
300ms   ->  Commited:   251,240
400ms   ->  Used:       6,272
400ms   ->  Commited:   251,240
500ms   ->  Used:       1,080,348,112
500ms   ->  Commited:   1,325,924,352
600ms   ->  Used:       1,112
600ms   ->  Commited:   1,352
700ms   ->  Used:       1,112
700ms   ->  Commited:   1,352
800ms   ->  Used:       1,112
800ms   ->  Commited:   1,352
900ms   ->  Used:       1,112
900ms   ->  Commited:   1,352
1000ms  ->  Used:       1,112
1000ms  ->  Commited:   1,352
1100ms  ->  Used:       1,112
1100ms  ->  Commited:   1,352
1200ms  ->  Used:       2,261,768
1200ms  ->  Commited:   1,352
1300ms  ->  Used:       2,768
1300ms  ->  Commited:   1,352

你可以看到,我的堆的初始大小约为260MB,允许的最大大小(GC可能决定调整大小的大小)大约为3.7 GB.

在创建阵列之前,使用了大约6MB的堆.然后创建大数组,我的堆大小(提交大小)增加到1,3GB,使用大约1GB(数组).然后我强制收集数组的垃圾收集.然而,我的堆大小保持在1,因为GC不关心再次收缩,只是利用率下降到2MB.

– 结果G1:

Init:   262,000
Max:    4,179,623,936

0ms ->  Used:       2,097,152
0ms ->  Commited:   262,000
100ms   ->  Used:       2,152
100ms   ->  Commited:   262,000
200ms   ->  Used:       2,152
200ms   ->  Commited:   262,000
300ms   ->  Used:       2,152
300ms   ->  Commited:   262,000
400ms   ->  Used:       2,152
400ms   ->  Commited:   262,000
500ms   ->  Used:       1,074,364,464
500ms   ->  Commited:   1,336,934,400
600ms   ->  Used:       1,464
600ms   ->  Commited:   1,400
700ms   ->  Used:       1,464
700ms   ->  Commited:   1,400
800ms   ->  Used:       1,464
800ms   ->  Commited:   1,400
900ms   ->  Used:       1,464
900ms   ->  Commited:   1,400
1000ms  ->  Used:       492,520
1000ms  ->  Commited:   8,388,608
1100ms  ->  Used:       492,520
1100ms  ->  Commited:   8,608
1200ms  ->  Used:       492,520
1200ms  ->  Commited:   8,608

现在我们开始! G1 GC关心小堆!清理对象后,不仅利用率降至约0.5MB,而且堆大小也缩小到8MB(与ParallelOldGC中的1,3GB相比)

更多信息:

但是,请记住,堆大小仍将与任务管理器中显示的不同.从Wikipedia – Java virtual machine开始的following image说明堆只是完整JVM内存的一部分:

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读