-XX:PretenureSizeThreshold的默认值和作用浅析

简介: -XX:PretenureSizeThreshold的默认值和作用浅析

一、背景

《深入理解Java虚拟机》第93页,3.6.2 大对象直接进入老年代。

讲到大对象主要指字符串和数组,虚拟机提供了一个-XX:PretenureSizeThreshold参数,大于这个值的参数直接在老年代分配。

这样做的目的是避免在Eden区和两个Survivor区之间发生大量的内存复制(新生代采用复制算法)。

但是这里没讲清楚默认值是多少,默认会不会“大”对象直接进入老年代。

二、解析

2.1 参考文章
找到了一篇相关问题的文章《Frequently Asked Questions about Garbage Collection in the HotspotTM JavaTM Virtual Machine》

第29条:Do objects ever get allocated directly into the old generation?

In 1.4.1 there two situations where allocation may occur directly into the old generation.

有两种情况,对象会直接分配到老年代。
If an allocation fails in the young generation and the object is a large array that does not contain any references to objects, it can be allocated directly into the old generation. In some select instances, this strategy was intended to avoid a collection of the young generation by allocating from the old generation.

如果在新生代分配失败且对象是一个不含任何对象引用的大数组,可被直接分配到老年代。

通过在老年代的分配避免新生代的一次垃圾回收。

There is a flag (available in 1.4.2 and later) l-XX:PretenureSizeThreshold= that can be set to limit the size of allocations in the young generation. Any allocation larger than this will not be attempted in the young generation and so will be allocated out of the old generation.

XX:PretenureSizeThreshold=<字节大小>可以设分配到新生代对象的大小限制。

任何比这个大的对象都不会尝试在新生代分配,将在老年代分配内存。

The threshold size for 1) is 64k words. The default size for PretenureSizeThreshold is 0 which says that any size can be allocated in the young generation.

PretenureSizeThreshold 默认值是0,意味着任何对象都会现在新生代分配内存。

2.2 实验解析
设置虚拟机参数

-Xms2048m
-Xmx2048m
-Xmn1024m
-XX:+UseConcMarkSweepGC

-XX:SurvivorRatio=8

-Xms表示初始化堆内存

-Xmx 表示最大堆内存

-Xmn表示新生代的内存

-XX:SurvivorRatio=8表示新生代的Eden占8/10,S1和S2各占1/10.

因此Eden的内存大小为:0.8102410241024字节 约为81910241024

上代码

public class Test {

    public static void main(String[] args) throws Exception {

        byte[] array = new byte[700 * 1024 * 1024];//734003216

        for (MemoryPoolMXBean memoryPoolMXBean : ManagementFactory.getMemoryPoolMXBeans()) {

            System.out.println(memoryPoolMXBean.getName() + "  总量:" + memoryPoolMXBean.getUsage().getCommitted() + "   使用的内存:" + memoryPoolMXBean.getUsage().getUsed());
        }

    }
}

可以看到应该被分配到了新生代的Eden区

Code Cache  总量:2555904   使用的内存:1206528
Metaspace  总量:4980736   使用的内存:3426872
Compressed Class Space  总量:524288   使用的内存:371280
Par Eden Space  总量:859045888   使用的内存:785546016
Par Survivor Space  总量:107347968   使用的内存:0
CMS Old Gen  总量:1073741824   使用的内存:0

将数组对象增加到大于Eden区内存大小

byte[] array = new byte[900 1024 1024];
会导致新生代分配失败,直接进入老年代

Code Cache  总量:2555904   使用的内存:1197824
Metaspace  总量:4980736   使用的内存:3426024
Compressed Class Space  总量:524288   使用的内存:371280
Par Eden Space  总量:859045888   使用的内存:34361864
Par Survivor Space  总量:107347968   使用的内存:0
CMS Old Gen  总量:1073741824   使用的内存:943718416

且此时通过jstat -gcutil vmpid命令查看垃圾回收状态,发现并没有YGC.

然后我们将字节数组改回 小于Eden去内存

byte[] array = new byte[700 1024 1024];

并添加启动参数-XX:PretenureSizeThreshold=100000000

Code Cache  总量:2555904   使用的内存:1172160
Metaspace  总量:4980736   使用的内存:3412552
Compressed Class Space  总量:524288   使用的内存:371280
Par Eden Space  总量:859045888   使用的内存:51542800
Par Survivor Space  总量:107347968   使用的内存:0
CMS Old Gen  总量:1073741824   使用的内存:734003216

发现即使新生代足够分配,大于这个值的大对象也直接在老年代分配。

从而印证了上面文档的说法。

三、总结

多查权威参考文档,参考价值更大。

多试验看效果,实践印象才更加深刻。

思路参考自:https://www.jianshu.com/p/f7cde625d849

相关文章
|
2月前
如何处理构造函数中参数的默认值?
设置合理的默认值可以增加代码的灵活性和易用性,同时减少在调用构造函数时必须传递所有参数的要求。在处理默认值时,要确保其合理性和一致性,避免出现意外的行为或错误。你还想了解关于构造函数的其他方面吗?比如参数的验证等
43 1
|
数据库
设置EntityFramework中decimal类型数据精度问题(EF默认将只会保留到2为精度)
原文:设置EntityFramework中decimal类型数据精度 EF中默认的decimal数据精度为两位数,当我们数据库设置的精度大于2时,EF将只会保留到2为精度。 e.g. 2.19990将会被保存为2.
3029 0
|
8月前
|
JavaScript 前端开发
分享函数中 this 在不同场景下的指向与默认值,动态指定函数 this 值的方法
分享函数中 this 在不同场景下的指向与默认值,动态指定函数 this 值的方法
|
Linux Shell C语言
SO_RCVBUF和SO_SNDBUF默认值
SO_RCVBUF和SO_SNDBUF默认值
160 0
|
前端开发
前端学习案例-参数默认值是函数2
前端学习案例-参数默认值是函数2
96 0
前端学习案例-参数默认值是函数2
|
前端开发
前端学习案例-参数默认值是函数1
前端学习案例-参数默认值是函数1
70 0
前端学习案例-参数默认值是函数1
JSONModel 设置所有属性为可选
JSONModel 设置所有属性为可选
62 0
|
PHP 开发者
参数默认值|学习笔记
快速学习参数默认值
参数默认值|学习笔记
|
测试技术 Android开发
怎么知道方法的参数有哪些类型呢?
怎么知道方法的参数有哪些类型呢?
224 0

热门文章

最新文章