Java ParNew Collection 将对象移动到老年代

标签 java object garbage-collection

这是我的虚拟机参数:

-XX:MaxPermSize=128m 
-XX:+UseParNewGC 
-XX:MaxNewSize=1G 
-XX:NewSize=1G 
-Xms13G 
-Xmx13G 
-XX:SurvivorRatio=128 
-XX:MaxTenuringThreshold=0 
-XX:+UseTLAB 
-XX:+UseConcMarkSweepGC 
-verbose:gc 
-XX:+PrintGCDetails 
-XX:+PrintGCTimeStamps 
-Xloggc:./gc.log -cp

我还造成了内存泄漏:

{
    String s = "";
    for (int i = 0; i < objectsCount; i++)
        s += "s" + s;
}//Here all created objects should be garbage collected

所以我已经运行了这部分代码几次,我注意到大多数对象都被移动到老一代,只有一些对象被小型 GC 收集

832.135: [GC 832.135: [ParNew: 1032095K->0K(1040512K), 0.7309045 secs] 4832264K->4761739K(13623424K), 0.7309703 secs] [Times: user=2.37 sys=0.33, real=0.73 secs] 
833.148: [GC 833.148: [ParNew: 826257K->0K(1040512K), 0.1510836 secs] 5587996K->5095138K(13623424K), 0.1511436 secs] [Times: user=0.33 sys=0.00, real=0.15 secs] 
833.567: [GC 833.567: [ParNew: 742459K->0K(1040512K), 0.2067575 secs] 5837597K->5836296K(13623424K), 0.2068291 secs] [Times: user=0.75 sys=0.00, real=0.21 secs] 
853.490: [GC 853.582: [ParNew: 993475K->0K(1040512K), 0.6963577 secs] 8100247K->7107608K(13623424K), 0.6968597 secs] [Times: user=1.61 sys=0.00, real=0.79 secs] 
879.973: [GC 879.973: [ParNew: 1032433K->0K(1040512K), 10.6403908 secs] 8140042K->8061725K(13623424K), 10.6404844 secs] [Times: user=30.93 sys=0.34, real=10.64 secs] 

我认为原因是,太多的对象同时粘贴到内存中,因此垃圾收集器更方便地将它们从年轻代或老一代中移动。

我的问题是,我应该改变我的参数,以使所有这些对象都被 Small GC 清除?

最佳答案

将 String s = ""移到循环内,然后尝试。

这里的问题是,在 for 循环完成之前,变量 s 不会被标记为 GC。您看到的 GC 日志条目是在 for 循环进行时的。因为程序耗尽了年轻代空间,并且 for 循环仍在运行,所以 GC 收集器将 S 的值移动到老代,因为它不知道还能做什么。

<小时/>

而不是这样做

String s = "";
for(int i = 0; i < counter; i++){
    s = s + "something";
}

这样做

for(int i = 0; i < counter; i++){
    String s = "something";
    //call some methods to process it.
}

或者使用 StringBuilder 创建字符串连接,而不是在字符串上使用 + 运算符。

<小时/>
-d64 -server -Xms4096m -Xmx4096m -XX:PermSize=512m -XX:MaxPermSize=512m -XX:NewSize=1536m -XX:MaxNewSize=1536m -XX:-UseAdaptiveSizePolicy -XX:+UseParNewGC -XX:+CMSParallelRemarkEnabled -XX:+UseTLAB -XX:+UseConcMarkSweepGC -XX:+OptimizeStringConcat -XX:ParallelGCThreads=14 -XX:ConcGCThreads=14

根据需要调整烫发大小。 NewSize 是您的年轻一代,剩余的堆分配将分配给老一代。尝试一下这个,看看它是否对你有帮助。

关于Java ParNew Collection 将对象移动到老年代,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20314811/

相关文章:

java - 如何创建从数据库获取并保存文件夹的图片的路径?

javascript - 从 selenium webdriver 中的 flash 播放器读取内容

java - 我有很多分析调用要发送,如果缓冲区未满,如何缓冲它们并在一定时间内刷新

java - Java类型参数方法和构造函数问题

Javascript对象多次循环,顺序相同?

java - 在 Android 中,我可以在没有 Intent 的情况下在 Activity 之间移动对象吗?

java - 使用 Java 中的 Book 类和 TestBook 类创建程序

android - 此 java 代码是否会导致垃圾收集问题,如果是,我该如何预防?

c# - 空引用类型变量会影响垃圾收集器 (GC) 性能吗?

Java 优化 : Local Variable Vs Instance Variable