栈上分配
- 将线程独有的对象打散分配在Java栈上,而不是堆上;栈上的对象随着方法调用的结束而销毁,继而不需要GC便可回收内存空间;
- 逃逸分析:分析哪些对象是线程独有的;
- 标量替换:将对象的字段当成局部变量直接分配在Java栈中;
栈上分配示例
参数设置
- -XX:+DoEscapeAnalysis 开启逃逸分析,JDK8默认开启;
- -XX:+EliminateAllocations 开启标量替换,JDK8默认开启;
- -XX:-UseTLAB 禁用TLAB,,JDK8默认开启;
说明
- 在10m的堆上分配1亿个对象,势必造成大量GC,如果开启逃逸分析和标量替换,对象不会分配到堆中,从而避免了大量GC开销;
- 运行时间差距:9ms - 21s;
public class OnStackTest {
public static class User {
public int id = 0;
public String name = "";
}
public static void alloc() {
User u = new User();
u.id = 5;
u.name = "xxx";
}
/**
-server
-XX:+DoEscapeAnalysis
-XX:+EliminateAllocations
-XX:-UseTLAB
-Xmx10m
-Xms10m
-XX:+PrintGC
* @param args
*/
public static void main(String[] args) {
long b = System.currentTimeMillis();
for (int i = 0; i < 1000000000; i++) {
alloc();
}
long e = System.currentTimeMillis();
System.out.println(e - b);
}
}
结果对比
开启栈上分配
8
关闭栈上分配
21992