根据Java虚拟机规范,JVM将内存划分为:
年轻代(New)、年老代(Tenured)、持久代(Perm)
年轻代(New)
年轻代用来存放JVM刚分配的Java对象。
年轻代分为三个区。一个Eden区,两个Survivor区。两个Survivor空间一样大,当Eden中的对象经过垃圾回收没有被回收掉时,会在两个Survivor之间来回Copy,当满足某个条件,比如Copy次数,就会被Copy到年老代。
年老代(Tenured)
年轻代中经过垃圾回收没有回收掉的对象将被Copy到年老代
持久代(Perm)
永久代存放Class、Method元信息,其大小跟项目的规模、类、方法的量有关,一般设置为128M就足够,设置原则是预留30%的空间。
Java对象的生命周期
新建Java对象出生在年轻代(New)的Eden区。当Eden区满的时候,就会copy到Survivor区,经过Survivor1区和Survivor2区的迭代之后,最终被copy到年老代(Tenured)
垃圾回收
GC有两种类型:普通 GC 和 Full GC
JVM给了三种选择:串行收集器、并行收集器、并发收集器,但是串行收集器只适用于小数据量的情况,所以这里的选择主要针对并行收集器和并发收集器
当年轻代内存满时,会引发一次普通GC,该GC仅回收年轻代。需要强调的是年轻代满是指Eden代满,Survivor满不会引发GC。
当年老代满时会引发Full GC,Full GC将会同时回收年轻代、年老代。
当持久代满时也会引发Full GC,会导致Class、Method元信息的卸载。
思路:调整各代的内存比例和垃圾回收算法,提高吞吐量
整个JVM内存大小=年轻代大小 + 年老代大小 + 持久代大小。
设置JVM内存大小
-Xms3550m:设置JVM初始内存为3550m。
-Xmx3550m:设置JVM最大可用内存为3550m。
注意:很多人会把这两个值设置的不一样,其实这是有问题的。
这两个值设置相同,以避免每次垃圾回收完成后JVM重新分配内存。
设置年轻代,年老代比例
-XX:NewSize2g:设置年轻代的初始大小
-XX:MaxNewSize2g:设置年轻代的最大值
注意:这两个值设置相同,两个参数组合相当于下面这个参数。
-Xmn2g:设置年轻代的最大和最小都为2G。
注意:此值对系统性能影响较大,一般不设置。Sun官方推荐配置为整个堆的3/8。
-XX:NewRatio=4:设置年轻代和年老代的比值。
注意:推荐设置为4,一般不设置
如:为4,表示年轻代与年老代比值为1:4,年轻代占整个堆内存的1/5
注意:在设置了-XX:MaxNewSize的情况下,-XX:NewRatio的值会被忽略
老年代的内存=堆内存 - 新生代内存。老年代的最大内存 = 堆内存 - 新生代 最大内存。
-XX:SurvivorRatio=4:设置年轻代中Eden区与Survivor区的大小比值。
注意:Survivor区有两个。性能影响较大,一般不设置。
如:设置为4,则两个Survivor区与一个Eden区的比值为2:4,一个Survivor区占整个年轻代的1/6
设置每个线程的堆栈大小:
-Xss1024k:JDK5.0以后每个线程堆栈大小为1M,以前每个线程堆栈大小为256k。
注意:一般不设置,或者根据应用的线程所需内存大小进行调整
设置持久代大小:
-XX:PermSize=256m:设置持久代初始值
-XX:MaxPermSize=256m:设置持久代最大值
注意:推荐两个值相同,256m。
GC配置(推荐吞吐量优先的并行收集器)
-XX:+UseParallelOldGC:配置年老代垃圾收集方式为并行收集。
-XX:+UseParallelGC:配置年轻代垃圾收集方式为并行收集。
HeapDump JVM信息
-XX:+HeapDumpOnOutOfMemoryError
-XX:HeapDumpPath=/usr/aaa/dump
-XX:+PrintGCDetails
-XX:+PrintGCTimeStamps
-Xloggc:/usr/aaa/dump/heap_trace.txt