对象逃逸分析和标量替换
对象逃逸分析和标量替换都是对象在栈上分配问题
对象在栈上分配
通过JVM的内存分配,我们可以知道对象是在堆上分配的,但是当对象没有被引用的时候,需要靠GC进行回收,当对象较多的时候,会给GC带来较大的压力,也影响了性能,为了减少临时对象在堆内存分配的压力,JVM通过逃逸分析确定该对象不会被外部访问,也就是不会逃逸,于是在栈上分配内存,这样对象所占用的空间就能随着栈帧的销毁而销毁,减轻了垃圾回收的压力。
对象逃逸分析:就是分析对象动态作用域,当一个对象在方法中被定义后,它可能被外部方法所引用,例如作为调用参数传递到其他地方中。JDK7 默认开启
但是我们都知道,分配对象的内存是连续的,但是可能栈帧中有存放对象的大小,但是它不连续,所以这时候就引出了标量替换。
标量替换:当通过逃逸分析,分析该对象没有被外部访问时,JVM不会创建对象,而是将这个对象的成员变量分解成被这个方法所使用的成员变量所代替,这些代替成员变量在栈帧中或寄存器分配空间,这样就不会因为没有一块连续空间对象内存不够分配了。JDK7 默认开启
名称 | 关闭 | 开启 |
---|---|---|
开启逃逸分析 | -XX:-DoEscapeAnalysis | -XX:+DoEscapeAnalysis |
标量替换 | -XX:-EliminateAllocations | -XX:+EliminateAllocations |
代码测试
首先,我的idea 使用的jdk1.8 所以呢,它说自动开启逃逸分析和标量替换的
public class AllotOnStack {
public static void main(String[] args) {
long start = System.currentTimeMillis();
for (int i = 0; i < 100000000; i++) {
alloc();
}
long end = System.currentTimeMillis();
System.out.println(end - start);
}
private static void alloc() {
//如果开启逃逸分析和标量替换 会分配上栈上面,不会产生GC
User user = new User();
user.setId(1);
user.setName("zhuge");
}
}
下面再idea的VM option 添加参数
-Xmx 是最大堆的大小
-Xms 最小堆的大小
-XX:+PrintGC :打印GC 日志
使用如下参数不会发生GC 开启逃逸分析 标量替换
-Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:+EliminateAllocations
运行结果
标量替换的前提是逃逸分析
* 使用如下参数都会发生大量GC 关闭逃逸分析 开启标量替换
* -Xmx15m -Xms15m -XX:-DoEscapeAnalysis -XX:+PrintGC -XX:+EliminateAllocations
运行结果
//开启逃逸分析,没有开启标量替换 ,还是大量GC
-Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:-EliminateAllocations
运行结果
结论
栈上分配依赖于逃逸分析和标量替换