JVM 调优过程
-Xms 初始化堆大小(默认物理内存1/64,根据实际需求)
-Xmx 最大堆大小
整个堆大小=年轻代+年老代+持久代
一 jvm新生代(young generation)
刚创建的java对象会分配新生代里面
1)年轻代= 1个e区+2个s区
2)-Xmn 年轻代大小(1.4之后版本 or lator)
3)-XX:NewRatio (该参数一般不使用)
年轻代(e区和二个S区)
4) -XX:SurvivorRatio
e区与s区的大小比值,设置为8 ,则二个s区与一个e区比例:2:8
5)
二 java 老年代
1) 老年代= 整个堆-年轻代-持久代
2)年轻代中经过 垃圾回收没有回收对象被复制到年老代,
3)老年代存储对象比年轻代年龄大的多,而且不乏大对象。(文件,缓存,数组)
对于大对象,可通过启动参数设置(-XX:PretenureSizeThreshold=1024)单位字节也就是1M,
默认为0来代表超过多大就不在新生代分配。(建议不需要分配)
三 java 持久代(perm generation)
1) 持久代 = 整个堆 - 年轻代 - 老年代大小
2) -XX:PermSize -XX:MaxPermSize
设置持久代的大小,一般情况推荐把-XX:PermSize设置-XX:MaxPermSize相同,
因为持久代大小的调整也会导致堆内存触发一次Fgc
3) 持久代里面存放Class,Methmod元信息,其大小与项目的规模,类,方法的数量有关,
一般设置为128M就足够,保留30%的空间。
4) 持久代的回收方式
常量池中的常量,无用的类信息,常量的回收很简单,没有引用就可以回收了
对于无用的类进行回收,必须保证3点:
类的所有实例都已经被回收
加载类的classLoader已经被回收
jvm 垃圾收集算法
1. 引用计数算法(已经不用了,很早版本)
2. 根搜索算法
jvm 垃圾回收算法
1. 复制算法(copying)
abc三个对象,进入e区的时候,把需要的复制到s0,s1区,没有被引用直接被回收了。(gcroot)根扫描
新生代中二个s0区,就应用到此场景
2. 标记清除算法(Mark-sweep)
适合老年代回收,不需要对象移动。
3. 标记整理压缩算法(Mark-Compac)
对象进行移动,会将存活的对象往左端,标记整理标记算法是在标记清除算法的基础上,又进行对象移动,因此
成本更高,但是却解决了内存碎片的问题。
名词解释
1 . 串行回收
gc单线程内存回收,会暂停所有用户线程
2 . 并行回收
收集是指多个GC线程并发工作,但此时用户线程是暂停;所以,serial是串行的,
parallel收集是并行的,而CMS收集器是并发的。
3 . 并发回收
是指用户线程与GC线程同时执行(不一定是并行,可以交替,但总体同时执行),不需要
要停顿用户线程(其实在CMS中用户线程还是需要停顿的,只是非常短,GC线程在另一个
CPU上执行);
-XX:+UseConcMarkSweepGC 表示使用CMS
-XX:+CMSParallelRemarkEnabled 表示并行remark
-XX:+UseCMSCompactAtFullCollection 表示在FGC之后进行压缩,因为CMS默认不压缩空间的。
-XX:+UseCMSInitiatingOccupancyOnly 表示只在到达阀值的时候,才进行CMS GC
-XX:CMSInitiatingOccupancyFraction=70 设置阀值为70%,默认为68%。