内存模型
JVM的内存模型大致为如下图所示
JVM对内存采用的是分代的方式进行管理,而采用这种方式的原因:不同对象的生命周期是不一样的,不同生命周期的对象可以采取不同的方式进行管理,这样效率更高。JVM按代进行划分,方法区为持久代,堆为老年代和新生代,大致如下图所示
年轻代:新生成的对象首先都是存放到年轻代,年轻代的目标是尽可能快速的回收掉那些生命周期短的对象。年轻代分3个区:Eden去,两个Survivor区。大部分对象在Eden区中生成,当Eden区满时,还存活的对象将被复制到Survivor区,在Survivor区经过N次还存活的对象,将被放到老年代。
老年代:存放的都是一些生命周期较长的对象。
持久代:用于存放已加载的类信息、常量、静态变量以及方法代码。这里需要注意,如果程序中有动态生成或者动态调用的class时,此时最好将持久代的空间设置的大些。
JVM给每一个线程都会分配一个java栈,且线程结束后,栈就会被回收。栈的大小是固定的,栈本身是由一个一个的栈帧组成的先进先出的栈式结构,每个栈帧中存放方法运行时产生的局部变量、方法出口等信息,当调用一个方法时,栈就会创建一个栈帧存放这些数据,当方法调用完成时,栈帧消失,如果方法中调用了其他方法,则继续在栈顶创建新的栈帧。栈的生命周期和线程的生命周期一样,上面说JVM对内存的管理采用分代的方式,这些“内存”指的是“自己”只管创建,不管回收的内存,而栈是“我”创建,我回收,所以,在上面的分代管理中没有栈。
本地方法栈的管理也是如java栈一样。
参数配置
-Xmx=2048m:设置JVM最大可用的堆空间为2048m。
-Xms=2048m:初始化JVM的堆空间大小为2048m。如果-Xms=10m,初始化时,JVM向操作系统申请的堆空间大小10m,随着程序的运行,JVM可能会向操作系统继续申请内存,直到2048m,当JVM进行GC后,JVM可能重新分配内存,即可能将空余内存归还操作系统,JVM向操作系统申请或归还内存是消费资源的,所以,比较好的方式是,将-Xmx和-Xms的大小设置成一样。
-Xmn=1g:设置年轻代大小为1G,Sun官方推荐配置为整个堆的3/8。-XX:NewSize表示设置年轻代的初始值,-XX:MaxNewSize表示设置年轻代的最大值,-Xmn就是将-XX:NewSize和-XX:MaxNewSize的值设置成一样,-Xmn的优先级较低。-XX:NewRatio表示年轻代和年老代的比值,优先级低于-Xmn。配置时推荐使用-Xmn参数。
-XX:SurvivorRatio=8:设置年轻代中Eden区与Survivor区的大小比值。设置为4表示,Eden:Survivor(一个)=1:8,因年轻代有两个一样的Survivor区,所以,Eden:Survivor=8:2,即一个Survivor占新生代空间的1/10,官方推荐幸存代占新生代的1/10。
-Xss=128k:设置每个线程的栈的空间,JDK5.0以后的默认大小为1m,以前为256k,在相同物理内存下,减少这个值能生成更多的线程,但是,操作系统对于一个进程内的线程数还是有限制的,不能无限生成,经验值在3000~5000左右。
-XX:PermSize=128m:设置初始化持久代的空间为128m,一般固定大小为64m。
-XX:MaxPermSize=256m:持久代的最大可申请的空间为256m。