1. 什么是逃逸分析
逃逸分析(Escape Analysis)是目前Java虚拟机中比较前沿的优化技术
。
逃逸分析的基本原理是:分析对象动态作用域,当一个对象在方法里面被定义后,它可能被外部方法所引用
,例如作为调用参数传递到其他方法中,这种称为方法逃逸;甚至还有可能被外部线程访问到,譬如赋值给可以在其他线程中访问的实例变量,这种称为线程逃逸;从不逃逸、方法逃逸到线程逃逸,称为对象由低到高的不同逃逸程度。
简单来说,如果对象的生命周期仅在方法内部,方法结束,该变量就消耗,那么就叫不逃逸;反之,是逃逸
下面我们分别给出t逃逸和非逃逸的例子:
//逃逸
public User test1() {
User user = new User();
user.setId(1);
user.setName("zhuge");
// TODO 保存到数据库
return user; //作为返回值,继续被外部使用
}
//非逃逸
public void test2() {
User user = new User();
user.setId(1);
user.setName("zhuge");
// TODO 保存到数据库
}
2. 作用
我们通过JVM内存分配可以知道JAVA中的对象都是在堆上进行分配,当对象没有被引用的时候,需要依靠GC进行回收内存,如果对象数量较多的时候,会给GC带来较大压力,也间接影响了应用的性能。
为了减少临时对象在堆内分配的数量,JVM通过逃逸分析确定该对象不会被外部访问。如果不会逃逸可以将该对象在栈上分配内存,这样该对象所占用的内存空间就可以随栈帧出栈而销毁,就减轻了垃圾回收的压力。
3. 栈上分配
如果符合不逃逸条件,那么可以采取优化方案,栈上分配
,而栈上分配依赖的以下手段:
-
标量替换(Scalar Replacement)
若一个数据已经无法再分解成更小的数据来表示了,Java虚拟机中的原始数据类型(int、long等数值类型及reference类型等)都不能再进一步分解了,那么这些数据就可以被称为标量
。相对的,如果一个数据可以继续分解,那它就被称为
聚合量
。标量与聚合量:标量即不可被进一步分解的量,而JAVA的基本数据类型就是标量(如:int,long等基本数据类型以及
reference类型等),标量的对立就是可以被进一步分解的量,而这种量称之为聚合量。而在JAVA中对象就是可以被进一
步分解的聚合量。通过逃逸分析确定该对象不会被外部访问,并且对象可以被进一步分解时,JVM不会创建该对象,而是
将该对象成员变量分解若干个被这个方法使用的成员变量所代替
,这些代替的成员变量在栈帧或寄存器上分配空间,这样就不会因为没有一大块连续空间导致对象内存不够分配。
4. 栈上分配示例
-XX:+DoEscapeAnalysis 手动开启逃逸分析
-XX:+EliminateAllocations 开启标量替换参数
JDK7之后默认开启逃逸分析,2个参数默认都开启
关闭的话,把加号替换为减号
/**
* 栈上分配,标量替换 代码调用了1亿次alloc(),如果是分配到堆上,大概需要1GB以上堆空间,如果堆空间小于该值,必然会触发GC。
*
* 使用如下参数不会发生GC -Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:+EliminateAllocations
* 使用如下参数都会发生大量GC -Xmx15m -Xms15m -XX:-DoEscapeAnalysis -XX:+PrintGC -XX:+EliminateAllocations
* -Xmx15m -Xms15m -XX:+DoEscapeAnalysis -XX:+PrintGC -XX:-EliminateAllocations
*/
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() {
User user = new User();
user.setId(1);
user.setName("zhangsan");
}
}
class User {
private int id;
private String name;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
使用注释头中的参数,如果开启基本上不打印gc日志,如果未开启,则频繁打印gc