目录 

一、逃逸分析的原理

二、编译器优化

三、Java伪代码

一、逃逸分析的原理

逃逸分析(Escape Analysis)是目前Java虚拟机中比较前沿的优化技术,它与类型继承关系分析一样,并不是直接优化代码的手段,而是为其他优化措施提供依据的分析技术。

逃逸分析的基本原理

  1. 方法逃逸:分析对象动态作用域,当一个对象在方法里面被定义后,它可能被外部方法所引用,例如作为调用参数传递到其他方法中,这种称为方法逃逸
  2. 线程逃逸:甚至还有可能被外部线程访问到,譬如赋值给可以在其他线程中访问的实例变量,这种称为线程逃逸

从不逃逸、方法逃逸到线程逃逸,称为对象由低到高的不同逃逸程度。

二、编译器优化

如果能证明一个对象不会逃逸到方法或线程之外(换句话说是别的方法或线程无法通过任何途径访问到这个对象),或者逃逸程度比较低(只逃逸出方法而不会逃逸出线程),则可能为这个对象实例采取不同程度的优化:

  1. 栈上分配(Stack Allocations):虚拟机的垃圾收集子系统会回收堆中不再使用的对象,但回收动作无论是标记筛选出可回收对象,还是回收和整理内存,都需要耗费大量资源。如果确定一个对象不会逃逸出线程之外,那让这个对象在栈上分配内存将会是一个很不错的主意,对象所占用的内存空间就可以随栈帧出栈而销毁,栈上分配可以支持方法逃逸,但不能支持线程逃逸。
  2. 标量替换(Scalar Replacement):
  1. 标量:若一个数据已经无法再分解成更小的数据来表示了,Java虚拟机中的原始数据类型(int、long等数值类型及reference类型等)都不能再进一步分解了,那么这些数据就可以被称为标量.
  2. 聚合量(Aggregate):相对的,如果一个数据可以继续分解,那它就被称为聚合量(Aggregate),Java中的对象就是典型的聚合量,Java中的对象就是典型的聚合量
  3. 标量替换:如果把一个Java对象拆散,根据程序访问的情况,将其用到的成员变量恢复为原始类型来访问,这个过程就称为标量替换
  4. 逃逸分析应用:假如逃逸分析能够证明一个对象不会被方法外部访问,并且这个对象可以被拆散,那么程序真正执行的时候将可能不去创建这个对象,而改为直接创建它的若干个被这个方法使用的成员变量来代替。将对象拆分后,除了可以让对象的成员变量在栈上(栈上存储的数据,很大机会被虚拟机分配至物理机器的高速寄存器中存储)分配和读写之外,还可以为后续进一步的优化手段创建条件
  5. 特点栈上分配的一种特例,实现更简单(不用考虑整个对象完整结构的分配),但对逃逸程度的要求更高,它不允许对象逃逸出方法范围内
  1. 同步消除(Synchronization Elimination):线程同步本身是一个相对耗时的过程,如果逃逸分析能够确定一个变量不会逃逸出线程,无法被其他线程访问,那么这个变量的读写肯定就不会有竞争,对这个变量实施的同步措施也就可以安全地消除掉

三、Java伪代码

1、未优化代码:

Java对象逃逸分析private算逃逸吗 java方法逃逸_标量

2、将Point的构造函数和getX()方法进行内联优化 

Java对象逃逸分析private算逃逸吗 java方法逃逸_Java_02

3、逃逸分析 :

,经过逃逸分析,发现在整个test()方法的范围内Point对象实例不会发生任何程度的逃逸,这样可以对它进行标量替换优化,把其内部的x和y直接置换出来,分解为test()方法内的局部变量,从而避免Point对象实例被实际创建,优化后的结果如下所示:

Java对象逃逸分析private算逃逸吗 java方法逃逸_标量_03

4、通过数据流分析,发现py的值其实对方法不会造成任何影响,那就可以放心地去做无效代码消除得到最终优化结果,如下所示:

Java对象逃逸分析private算逃逸吗 java方法逃逸_标量_04

四、逃逸分析相关参数

 JDK 7时这项优化才成为服务端编译器默认开启的选项,

    • -XX:+DoEscapeAnalysis:手动开启逃逸分析 
    • -XX:+PrintEscapeAnalysis:查看分析结果
    • -XX:+EliminateAllocations:开启标量替换
    • +XX:+EliminateLocks:开启同步消除
    • -XX:+PrintEliminateAllocations:查看标量的替换情况

    尽管目前逃逸分析技术仍在发展之中,未完全成熟,但它是即时编译器优化技术的一个重要前进方向,在日后的Java虚拟机中,逃逸分析技术肯定会支撑起一系列更实用、有效的优化技术。

    参考:《深入理解Java虚拟机:JVM高级特性与最佳实践第三版》