一、哪些是GC Roots ?
- 在虚拟机栈(栈帧中的本地变量表)中引用的对象,譬如各个线程被调用的方法堆栈中使用到的参数、局部变量、临时变量
- 在方法区中常量引用的对象,譬如字符串常量池(String Table)里的引用
- 在方法区中类静态属性引用的对象,譬如java类的引用类型静态变量
- 在本地方法栈中JNI(即通常所说的的native方法)引用的对象
- Java虚拟机内部的引用,如基本数据对应的Class对象,一些常驻的异常对象(比如NullpointException、outofMemoryError)等,还有系统类加载器
- 所有被同步锁(synchronized关键字)持有的对象
- 反应java虚拟机内部情况的JMXBean、JVMTI中注册的回调、本地代码缓存等
二、被引用的对象就一定能存活吗?
不一定,看Reference类型,弱引用在GC时会被回收,软引用在内存不足的时候,即OOM前会被回收,但如果没有在Reference chain 中的对象就一定会 被回收
三、强、软、弱、虚引用是什么?有什么区别?
1、强引用:就是普通对象引用关系,如String s =new String(*ConstXiong*)
2、软引用:用于维护一些可有可无的对象,只有在OOM之前会被回收,如果回收了软引用对象之后还是内存不足,则会抛出内存溢出异常,SoftReference实现
3、弱引用:相比软引用来说,更加无用一下,它拥有更短的生命周期,当JVM 进行垃圾回收时、无论内存是否充足,都会回收弱引用关联的对象,weakReference实现
4、虚引用:一种形同虚设的引用,在现实场景中用的不是很多,它主要用来跟踪对象被垃圾回收的活动,RhantomReference实现
四、你做过JVM调优,说说如何查看JVM 参数默认值?
1、jps -v 可以查看jvm 进程显示指定的参数
2、使用 -XX:+PrintFlagsFinal可以查看JVM所有参数的值
3、jinfo 可以实时查看和调整虚拟机各项参数
五、工作中常用的JVM参数有哪些?
java8为例
日志
1、-XX:+PrintFlagFinal,打印JVM所有参数的值
2、-XX:+PrintGC, 打印GC信息
3、-XX:+PrintGCDetails,打印GC详细信息
4、-XX:+PrintGCTimeStamps,打印GC时间戳
5、-Xloggc:filename,设置GC log 文件位置
6、-XX:+PrintTenuringDistribution,查看熬过收集后剩余对象的年龄分布信息
内存设置
1、-Xms,设置堆的初始化大小
2、-Xmx:设置堆的内存大小
3、-Xmn:设置新生代内存大小
4、-XSs:设置线程栈大小
5、-XX:NewRatio:新生代与老年代比值
6、-XX:SurvivoRatio,新生代中Eden区与两个Survivor区的比值,默认为8,即Eden:Survivor:Survivor = 8:1:1
7、-XX:MaxTenuringThreshold,从年轻代到老年代,最大晋升年龄,CMS下默认为6,G1下默认为15
8、-XX:MetaspaceSize:设置元空间大小。第一次超过将触发GC
9、-XX:MaxMetaspaceSize:元空间最大值
10、-XX:MaxDirectMemorySize,用于设置直接内存的最大值,限制通过DirectByteBuffer申请的内存
11、-XX:ReservedCodeCacheSize,用于设置JIT编译后的代码存放区大小,如果观察到这个值有限制,可以适当调大,一般够用即可
设置垃圾收集相关
1、-XX:+UserSerialGC .设置串行收集器
2、-XX:+UserParallelGC,设置并行收集器
3、-XX:+UserConcMarkSweepGC,使用CMS收集器
4、-XX:ParallelGCThreads,设置Parallel GC的线程数
5、-XXMaxGCPauseMillis,GC最大暂停时间ms
6、-XX:+UseG1GC,使用G1垃圾收集器
CMS垃圾回收器相关
1、+XX:+UseCMSInitiatingOccupancyOnly
2、+XX:CMSInitiatingOccupancyFraction ,与前者配合使用,指定MajorGC 的发生时机
3、-XX:+ExplicitGCInvokesConcurrent,代码调用System.gc()开始并行FullGC,建议加上这个参数
4、-XX:+CMSScavengeBeforeRemark,标示开启或关闭CMS重新标价阶段之前的清除(YGC)尝试,他可以降低remark 时间,建议加上
5、-XX:ParallelRefProcEnabled,可以并行处理Reference,以加快处理速度,缩短耗时
G1垃圾回收器相关
1、-XX:MaxGCPauseMillis,用于设置目标停顿时间,G1会尽力达成
2、-XX:G1HeapRegionSize,用于设置小堆区大小,建议保持默认
3、-XX:InitiatingHeapOccupancyPercent,表示当整个堆内存使用达到一定比例(默认是45%),并发标记阶段就会被启动
4、-XX:ConcGCThreads,表示并发垃圾收集器使用的线程数量,默认值随JVM运行的平台的不同而变动,不建议修改
六、谈谈对OOM的认识
1、除了程序计数器,其它内存区域都有OOM的风险
2、栈一般经常会发生*Error,比如32位的windows系统菜单进程限制2G内存,无限创建线程就会发生栈的OOM
3、java8常量池移动到堆中,溢出会出java.lang.OutOfMermoryError:java Heap space设置最大元空间大小参数无效
4、堆内存溢出,报错同上,这种比较好理解,GC之后无法在堆中申请内存创建对象就会报错
5、方法区OOM。经常会遇到的是动态生成大量的类,jsp等
6、直接内存OOM,涉及到-XX:MaxDirectMemorySize参数和Unsafe对象对内存的申请
七、什么情况发生栈溢出?
-Xss可以设置线程栈的大小,当线程方法递归调用层次太深或者栈帧中的局部变量过多时,会出现栈溢出错误java.lang.*Error
八、你有哪些手段来排查OOM的问题?
1、增加两个参数 -XX:HeapDumpOnOutOfMemoryError
-XX:heapDumpPath=/tmp/heapdump.hprof,当OOM发生时自动dump堆内存信息到指定目录
2、同时jstat查看监控JVM的内存和GC的情况,先观察问题大概率出现在什么区域
3、使用MAI工具载入到dump文件,分析大对象的占用情况,比如hashMap做缓存未清理,时间长了就会内存溢出,可以改为弱引用。