Java笔试&&面试经验Java学习笔记

面试小结之JVM篇

2017-06-13  本文已影响4742人  ginobefun

最近面试一些公司,被问到的关于Java虚拟机的问题,以及自己总结的回答。

Java内存区域是如何划分的?

Java内存区域

对象是否可用以及引用类型。

有哪些常见的垃圾收集算法?

PS:堆的划分及回收过程详解

1. Eden区最大,对外提供堆内存。当Eden区快要满了,则进行Minor GC,把存活对象放入Survivor A区,清空Eden区;
2. Eden区被清空后,继续对外提供堆内存;
3. 当Eden区再次被填满,此时对Eden区和Survivor A区同时进行Minor GC,把存活对象放入Survivor B区,同时清空Eden 区和Survivor A区;
4. Eden区继续对外提供堆内存,并重复上述过程,即在Eden区填满后,把Eden区和某个Survivor区的存活对象放到另一个Survivor区;
5. 当某个Survivor区被填满,且仍有对象未被复制完毕时或者某些对象在反复Survive 15次左右时,则把这部分剩余对象放到Old区;
6. 当Old区也被填满时,进行Major GC,对Old区进行垃圾回收。

有哪些常见的垃圾收集器?

这里讨论JDK 1.7 Update 14之后的HotSpot虚拟机,包含的虚拟机如下图所示(存在连线的表示可以搭配使用):

HotSpot垃圾收集器

Serial收集器

Serial收集器

ParNew收集器

ParNew收集器

PS:关于垃圾收集器的并行和并发

CMS收集器

CMS收集器

介绍一下G1收集器的原理和实现。

G1收集器
并行与并发(充分利用多核多CPU缩短STW时间)
分代收集(独立管理整个Java堆,但针对不同年龄的对象采取不同的策略)
空间整合(局部看是基于复制算法,从整体来看是基于标记-整理算法,都不会产生内存碎片)
可预测的停顿(可以明确指定在一个长度为M毫秒的时间片内垃圾收集不会超过N毫秒)
初始标记(标记一下GC Roots能直接关联的对象并修改TAMS值,需要STW但耗时很短)
并发标记(从GC Root从堆中对象进行可达性分析找存活的对象,耗时较长但可以与用户线程并发执行)
最终标记(为了修正并发标记期间产生变动的那一部分标记记录,这一期间的变化记录在Remembered 
Set Log里,然后合并到Remembered Set里,该阶段需要STW但是可并行执行)
筛选回收(对各个Region回收价值排序,根据用户期望的GC停顿时间制定回收计划来回收);

你们的服务配置的虚拟机参数是怎么样的?

我们的服务的虚拟机参数:

-server       --启用能够执行优化的编译器,显著提高服务器的性能
-Xmx4000M     --堆最大值
-Xms4000M     --堆初始大小
-Xmn600M      --年轻代大小
-XX:PermSize=200M         --持久代初始大小
-XX:MaxPermSize=200M      --持久代最大值
-Xss256K                  --每个线程的栈大小
-XX:+DisableExplicitGC    --关闭System.gc()
-XX:SurvivorRatio=1       --年轻代中Eden区与两个Survivor区的比值
-XX:+UseConcMarkSweepGC   --使用CMS内存收集
-XX:+UseParNewGC          --设置年轻代为并行收集
-XX:+CMSParallelRemarkEnabled        --降低标记停顿
-XX:+UseCMSCompactAtFullCollection   --在FULL GC的时候,对年老代进行压缩,可能会影响性能,但是可以消除碎片
-XX:CMSFullGCsBeforeCompaction=0     --此值设置运行多少次GC以后对内存空间进行压缩、整理
-XX:+CMSClassUnloadingEnabled        --回收动态生成的代理类 SEE:http://stackoverflow.com/questions/3334911/what-does-jvm-flag-cmsclassunloadingenabled-actually-do
-XX:LargePageSizeInBytes=128M        --内存页的大小不可设置过大, 会影响Perm的大小
-XX:+UseFastAccessorMethods          --原始类型的快速优化
-XX:+UseCMSInitiatingOccupancyOnly   --使用手动定义初始化定义开始CMS收集,禁止hostspot自行触发CMS GC
-XX:CMSInitiatingOccupancyFraction=80  --使用cms作为垃圾回收,使用80%后开始CMS收集
-XX:SoftRefLRUPolicyMSPerMB=0          --每兆堆空闲空间中SoftReference的存活时间
-XX:+PrintGCDetails                    --输出GC日志详情信息
-XX:+PrintGCApplicationStoppedTime     --输出垃圾回收期间程序暂停的时间
-Xloggc:$WEB_APP_HOME/.tomcat/logs/gc.log  --把相关日志信息记录到文件以便分析.
-XX:+HeapDumpOnOutOfMemoryError            --发生内存溢出时生成heapdump文件
-XX:HeapDumpPath=$WEB_APP_HOME/.tomcat/logs/heapdump.hprof  --heapdump文件地址

如何进行性能调优以及常用的JDK的命令行工具有哪些?

JDK的命令行工具

类的加载器是什么?

类加载器

谈谈你对Java内存模型的理解。

主内存与工作内存
1、程序次序规则:在一个单独的线程中,按照程序代码的执行流顺序,(时间上)先执行的操作happen—before(时间上)后执行的操作。
2、管理锁定规则:一个unlock操作happen—before后面(时间上的先后顺序,下同)对同一个锁的lock操作。
3、volatile变量规则:对一个volatile变量的写操作happen—before后面对该变量的读操作。
4、线程启动规则:Thread对象的start()方法happen—before此线程的每一个动作。
5、线程终止规则:线程的所有操作都happen—before对此线程的终止检测,可以通过Thread.join()方法结束、Thread.isAlive()的返回值等手段检测到线程已经终止执行。
6、线程中断规则:对线程interrupt()方法的调用happen—before发生于被中断线程的代码检测到中断时事件的发生。
7、对象终结规则:一个对象的初始化完成(构造函数执行结束)happen—before它的finalize()方法的开始。
8、传递性:如果操作A happen—before操作B,操作B happen—before操作C,那么可以得出A happen—before操作C。
public class Singleton {  
  
  private Singleton() {}  
  
  // Lazy initialization holder class idiom for static fields  
  private static class InstanceHolder {  
    private static final Singleton instance = new Singleton();  
  }  
  
  public static Singleton getSingleton() {   
    return InstanceHolder.instance;   
  }  
}  

是否了解偏向锁?

锁的优缺点对比

其他面试小结

扫一扫 关注我的微信公众号
上一篇 下一篇

猜你喜欢

热点阅读