linux下Java内存溢出排查

1、查询gc情况(每1秒钟打印一次gc情况)

jstat -gcutil pid 1000:1

查询结果含义:
S0:幸存区1占用率
S1:幸存区2占用率
E:Eden区占用率
O:老年区占用率
M:元数据区(java8,相当于java7及之前的永久代的概念)使用大小
ccs:压缩后使用率
YGC:young gc 次数,
YGCT:young gc耗时
FGC:full gc次数
FGCT:full gc耗时
GCt:GC共耗时

2、查询进程信息

#查询占用内存的进程(shift+m排序)
top
#存活的对象占用内存前100排序
jmap -histo:live 41843 | head -n 100

3、查询进程里面详细信息

jmap -heap 41843

如果Jave类的内存异常则检查代码
如果发现频繁的gc是因为新生代、老年代、永久代分配的大小有问题,则可以通过修改设置解决

eg:

java -Xmx3550m -Xms3550m -Xmn2g -Xss128k -XX:NewRatio=4 -XX:SurvivorRatio=4 -XX:PermSize=64M  -XX:MaxPermSize=128M -XX:MaxTenuringThreshold=0

参数含义:

  • -Xmx3550m 堆最大容量(heap max size)
  • -Xms3550m 堆最小容量(heap min size)
  • -Xmn2g 年轻代大小
  • -Xss256k 每个线程栈容量大小(stack size)
  • -XX:NewRatio=4 年轻代(包括Eden和两个Survivor区)与年老代的比值(除去持久代),设置为4,则年轻代与年老代所占比值为1:4,年轻代占整个堆栈的1/5;
  • -XX:SurvivorRatio=4 年轻代中Eden区与Survivor区的大小比值,设置为4,则两个Survivor区与一个Eden区的比值为2:4,一个Survivor区占整个年轻代的1/6
  • -XX:PermSize=64M 初始分配的永生代容量
  • -XX:MaxPermSize=128M 永生代最大容量
  • -XX:MaxTenuringThreshold=0 设置垃圾最大年龄

-Xmn对系统性能影响较大,Sun官方推荐配置为整个堆的3/8;JVM内存大小=年轻代大小 + 年老代大小 + 持久代大小持久代一般固定大小为64m,所以增大年轻代后,将会减小年老代大小。

JDK5.0以后每个线程堆栈大小为1M,以前每个线程堆栈大小为256K。更具应用的线程所需内存大小进行调整。在相同物理内存下,减小这个值能生成更多的线程。但是操作系统对一个进程内的线程数还是有限制的,不能无限生成,经验值在3000~5000左右;

-XX:MaxTenuringThreshold如果设置为0的话,则年轻代对象不经过Survivor区,直接进入年老代。对于年老代比较多的应用,可以提高效率。如果将此值设置为一个较大值,则年轻代对象会在Survivor区进行多次复制,这样可以增加对象再年轻代的存活时间,增加在年轻代即被回收的概论。

你可能感兴趣的:(Java)