(8)JVM的逃逸分析(上)

一、为什么“逃逸”

逃逸分析是指分析指针动态范围 的方法:同编译器优化原理的指针分析和外形分析相关联。当变量(或者对象)在方法中分配后,其指针有可能被返回或者被全局引用,就会其他方法或者线程所引用,称逃逸。    通俗讲,一个对象指针被多个方法或者线程引用时

public StringBuilder escapeDemo1(String a, String b) {

    StringBuilder stringBuilder = new StringBuilder();

    stringBuilder.append(a);

    stringBuilder.append(b);

    return stringBuilder;

}

stringBuilder是方法内部变量被直接返回,stringBuilder可能被其他地方的方法或参数所改变,作用域不只是demo1,虽是局部变量,但“逃逸”。

public String escapeDemo2(String a, String b) {

    StringBuilder stringBuilder = new StringBuilder();

    stringBuilder.append(a);

    stringBuilder.append(b);

    return stringBuilder.toString();//没有发生逃逸。

}

二、什么是逃逸分析

目的:有效减少Java 程序中  同步负载和内存堆分配压力  的跨函数全局数据流分析算法

Java Hotspot编译器分析出一个新的对象的引用使用范围从而决定是否要将这个对象分配到堆上。前沿优化技术了。

三、逃逸分析的原理

Java本身的限制(对象只能分配到堆中),为了减少临时对象在堆内分配的数量,方法体内定义局部变量,该变量在方法执行过程中未发生逃逸

    按照JVM调优机制,1)先内存创建,2)将对象引用压入调用,3)继续执行。

    用逃逸分析对JVM进行优化栈重新分配方式,1)先找未逃逸变量,2)将变量直接栈里无需进入堆,3)分配完,继续调用栈内执行,最后线程执行结束,栈空间被回收,局部变量也被回收了。优化前在堆中,优化后在栈中

四、逃逸的方式

  方法逃逸:对象跳出了方法(上面代码)

  线程逃逸:对象被其他线程访问赋值给实例变量,并被其他线程访问。对象逃出当前线程

五、逃逸分析的好处

  如果一个对象不会在方法体内,或线程内发生逃逸(或者说是通过逃逸分析后,使其未能发生逃逸)

1. 栈上分配

不逃逸的对象所占空间大,用栈空间,大量的对象将随方法的结束而销毁,减轻了GC压力

2. 同步消除

类的方法上有同步锁,运行时,只有一个线程在访问,此时逃逸分析后的机器码,会去掉同步锁运行

3. 标量替换

标量:不可分解原始数据类型(int,long等数值类型以及reference类型等)

聚合量可分解数据(对象)

如果逃逸分析证明一个可分解对象不会被外部访问,那程序真正执行的时候将可能不创建这个对象,而改为直接创建它的若干个被这个方法使用到的成员变量代替。拆散后的变量便可以被单独分析与优化,可以各自分别栈帧寄存器上分配空间,原本的对象就无需整体分配空间

六、开启设置

在JDK 6u23以上是默认开启,这里将设置重新明确一下: 

强制开启:1-server -XX:+DoEscapeAnalysis -XX:+PrintGCDetail -Xmx10m -Xms10m

关闭逃逸分析:1-server -XX:-DoEscapeAnalysis -XX:+PrintGCDetail -Xmx10m -Xms10m

写在结尾

栈空间一般小,无法存储大容量数据。目前的实现都是采用不那么准确但是时间压力相对较小的算法来完成逃逸分析,可能导致效果不稳定。逃逸分析的效果只能在满足高频和高数量的小容量的变量分配结构,才是合适的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 一、运行时数据区域 Java虚拟机管理的内存包括几个运行时数据内存:方法区、虚拟机栈、本地方法栈、堆、程序计数器,...
    luhanlin阅读 608评论 0 0
  •   我们都知道Java中的对象默认都是分配到堆上,在调用栈中,只保存了对象的指针。当对象不再使用后,需要依靠GC来...
    guoliang阅读 666评论 0 1
  • 第二部分 自动内存管理机制 第二章 java内存异常与内存溢出异常 运行数据区域 程序计数器:当前线程所执行的字节...
    小明oh阅读 1,279评论 0 2
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,896评论 0 11
  • 所有知识点已整理成app app下载地址 J2EE 部分: 1.Switch能否用string做参数? 在 Jav...
    侯蛋蛋_阅读 2,710评论 1 4

友情链接更多精彩内容