面试经历
记得几年前有一次阿里去面试,问到了这么一个问题:
Java中的对象都是在堆中分配吗?说明为什么!
当时我被问得一脸懵逼,瞬间被秒杀得体无完肤,当时我压根就不知道他在考什么知识点,难道对象不是在堆中分配吗?最后就没然后了,回去等通知了。
对象的分配
几乎所有的对象在堆中进行分配,这个是大家经常看到的一句话,但是这句话中并不意味着所有,JVM中对象是可以在栈中进行分配,但是前提是需要判断逃逸状态。
对象逃逸状态
1、全局逃逸(GlobalEscape)
即一个对象的作用范围逃出了当前方法或者当前线程,有以下几种场景:
对象是一个静态变量
对象是一个已经发生逃逸的对象
对象作为当前方法的返回值
2、参数逃逸(ArgEscape)
即一个对象被作为方法参数传递或者被参数引用,但在调用过程中不会发生全局逃逸,这个状态是通过被调方法的字节码确定的。
3、没有逃逸
即方法中的对象没有发生逃逸。
逃逸分析代码
publicclassEscapeAnalysisTest{publicstaticvoidmain(String[] args) throws Exception{longstart = System.currentTimeMillis();for(inti =0; i <50000000; i++) { allocate(); } System.out.println((System.currentTimeMillis() - start) +" ms"); Thread.sleep(600000); }staticvoidallocate(){ MyObject myObject =newMyObject(2020,2020.6); }staticclassMyObject{inta;doubleb; MyObject(inta,doubleb) {this.a = a;this.b = b; } }}
这段代码在调用的过程中 myboject这个对象属于全局逃逸,JVM可以做栈上分配
然后通过开启和关闭DoEscapeAnalysis开关观察不同。
开启逃逸分析(JVM默认开启)
查看执行速度
关闭逃逸分析
查看执行速度
测试结果可见,开启逃逸分析对代码的执行性能有很大的影响!那为什么有这个影响?
逃逸分析
如果是逃逸分析出来的对象可以在栈上分配的话,那么该对象的生命周期就跟随线程了,就不需要垃圾回收,如果是频繁的调用此方法则可以得到很大的性能提高。
采用了逃逸分析后,满足逃逸的对象在栈上分配
没有开启逃逸分析,对象都在堆上分配,会频繁触发垃圾回收(垃圾回收会影响系统性能),导致代码运行慢
代码验证
开启GC打印日志
-XX:+PrintGC
开启逃逸分析
可以看到没有GC日志
关闭逃逸分析
可以看到关闭了逃逸分析,JVM在频繁的进行垃圾回收(GC),正是这一块的操作导致性能有较大的差别。
总结
JVM是很多面试官非常喜欢问的。但是有些面试官的问题非常刁钻,因为 JVM 的知识体系博大精深,如果你没有把握就不要轻易回答。有些知识是需要实践才能体验到,心平气和的讲解这些知识点,会让你在面试中掌握主动地位。
网友评论