面试经历

记得几年前有一次阿里去面试,问到了这么一个问题:

Java中的对象都是在堆中分配吗?说明为什么!

当时我被问得一脸懵逼,瞬间被秒杀得体无完肤,当时我压根就不知道他在考什么知识点,难道对象不是在堆中分配吗?最后就没然后了,回去等通知了。

 

 

 

 

 

 

 

 

对象的分配

几乎所有的对象在堆中进行分配,这个是大家经常看到的一句话,但是这句话中并不意味着所有,JVM中对象是可以在栈中进行分配,但是前提是需要判断逃逸状态。

对象逃逸状态

1、全局逃逸(GlobalEscape)

即一个对象的作用范围逃出了当前方法或者当前线程,有以下几种场景:

对象是一个静态变量

对象是一个已经发生逃逸的对象

对象作为当前方法的返回值

2、参数逃逸(ArgEscape)

即一个对象被作为方法参数传递或者被参数引用,但在调用过程中不会发生全局逃逸,这个状态是通过被调方法的字节码确定的。

3、没有逃逸

即方法中的对象没有发生逃逸。

逃逸分析代码

public class EscapeAnalysisTest {
    public static void main(String[] args) throws Exception {
        long start = System.currentTimeMillis();
        for (int i = 0; i < 50000000; i++) {
            allocate();
        }
        System.out.println((System.currentTimeMillis() - start) + " ms");
        Thread.sleep(600000);
    }

    static void allocate() {
        MyObject myObject = new MyObject(2020, 2020.6);
    }

    static class MyObject {
        int a;
        double b;

        MyObject(int a, double b) {
            this.a = a;
            this.b = b;
        }
    }
}

这段代码在调用的过程中 myboject这个对象属于全局逃逸,JVM可以做栈上分配

然后通过开启和关闭DoEscapeAnalysis开关观察不同。

开启逃逸分析(JVM默认开启)

 

 查看执行速度

 

 关闭逃逸分析

 

 查看执行速度

 

 

测试结果可见,开启逃逸分析对代码的执行性能有很大的影响!那为什么有这个影响?

逃逸分析

如果是逃逸分析出来的对象可以在栈上分配的话,那么该对象的生命周期就跟随线程了,就不需要垃圾回收,如果是频繁的调用此方法则可以得到很大的性能提高。

采用了逃逸分析后,满足逃逸的对象在栈上分配

 

没有开启逃逸分析,对象都在堆上分配,会频繁触发垃圾回收(垃圾回收会影响系统性能),导致代码运行慢

 

 

代码验证

开启GC打印日志

-XX:+PrintGC

开启逃逸分析

 

 

可以看到没有GC日志

关闭逃逸分析

 

 

可以看到关闭了逃逸分析,JVM在频繁的进行垃圾回收(GC),正是这一块的操作导致性能有较大的差别。

总结

JVM是很多面试官非常喜欢问的。但是有些面试官的问题非常***钻,因为 JVM 的知识体系博大精深,如果你没有把握就不要轻易回答。有些知识是需要实践才能体验到,心平气和的讲解这些知识点,会让你在面试中掌握主动地位。



作者:让我来处理高并发
链接:https://www.jianshu.com/p/dd773ee22cb9
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。