JVM之逃逸分析

JVM之逃逸分析

作者:JAVA破局之路 2020-07-21 14:19:18

云计算

虚拟化 在编译程序优化理论中,逃逸分析是一种确定指针动态范围的方法——分析在程序的哪些地方可以访问到指针。它涉及到指针分析和形状分析。

[[334435]]

什么是逃逸分析

在编译程序优化理论中,逃逸分析是一种确定指针动态范围的方法——分析在程序的哪些地方可以访问到指针。它涉及到指针分析和形状分析。

当一个变量(或对象)在子程序中被分配时,一个指向变量的指针可能逃逸到其它执行线程中,或是返回到调用者子程序。如果使用尾递归优化(通常在函数编程语言中是需要的),对象也可以看作逃逸到被调用的子程序中。如果一种语言支持第一类型的延续性在Scheme和Standard ML of New Jersey中同样如此),部分调用栈也可能发生逃逸。

如果一个子程序分配一个对象并返回一个该对象的指针,该对象可能在程序中被访问到的地方无法确定——这样指针就成功“逃逸”了。如果指针存储在全局变量或者其它数据结构中,因为全局变量是可以在当前子程序之外访问的,此时指针也发生了逃逸。

逃逸分析确定某个指针可以存储的所有地方,以及确定能否保证指针的生命周期只在当前进程或在其它线程中。

下面我们看看Java中的逃逸分析是怎样的?

Java的逃逸分析只发在JIT的即时编译中,为什么不在前期的静态编译中就进行呢,知乎上已经有过这样的提问。

简单来说是可以的,但是Java的分离编译和动态加载使得前期的静态编译的逃逸分析比较困难或收益较少,所以目前Java的逃逸分析只发在JIT的即时编译中,因为收集到足够的运行数据JVM可以更好的判断对象是否发生了逃逸。关于JIT即时编译可参考JVM系列之走进JIT。

JVM判断新创建的对象是否逃逸的依据有:

一、对象被赋值给堆中对象的字段和类的静态变量。

二、对象被传进了不确定的代码中去运行。

如果满足了以上情况的任意一种,那这个对象JVM就会判定为逃逸。对于第一种情况,因为对象被放进堆中,则其它线程就可以对其进行访问,所以对象的使用情况,编译器就无法再进行追踪。第二种情况相当于JVM在解析普通的字节码的时候,如果没有发生JIT即时编译,编译器是不能事先完整知道这段代码会对对象做什么操作。保守一点,这个时候也只能把对象是当作是逃逸来处理。下面举几个例子

  1. public class EscapeTest { 
  2.  
  3.     public static Object globalVariableObject; 
  4.  
  5.     public Object instanceObject; 
  6.  
  7.     public void globalVariableEscape(){ 
  8.         globalVariableObject = new Object(); //静态变量,外部线程可见,发生逃逸 
  9.     } 
  10.  
  11.     public void instanceObjectEscape(){ 
  12.         instanceObject = new Object(); //赋值给堆中实例字段,外部线程可见,发生逃逸 
  13.     } 
  14.      
  15.     public Object returnObjectEscape(){ 
  16.         return new Object();  //返回实例,外部线程可见,发生逃逸 
  17.     } 
  18.  
  19.     public void noEscape(){ 
  20.         synchronized (new Object()){ 
  21.             //仅创建线程可见,对象无逃逸 
  22.         } 
  23.         Object noEscape = new Object();  //仅创建线程可见,对象无逃逸 
  24.     } 
  25.  

基于逃逸分析的优化

当判断出对象不发生逃逸时,编译器可以使用逃逸分析的结果作一些代码优化

将堆分配转化为栈分配。如果某个对象在子程序中被分配,并且指向该对象的指针永远不会逃逸,该对象就可以在分配在栈上,而不是在堆上。在有垃圾收集的语言中,这种优化可以降低垃圾收集器运行的频率。

同步消除。如果发现某个对象只能从一个线程可访问,那么在这个对象上的操作可以不需要同步。

分离对象或标量替换。如果某个对象的访问方式不要求该对象是一个连续的内存结构,那么对象的部分(或全部)可以不存储在内存,而是存储在CPU寄存器中。

对于优化一将堆分配转化为栈分配,这个优化也很好理解。下面以代码例子说明:

虚拟机配置参数:-XX:+PrintGC -Xms5M -Xmn5M -XX:+DoEscapeAnalysis

  • -XX:+DoEscapeAnalysis表示开启逃逸分析,JDK8是默认开启的
  • -XX:+PrintGC 表示打印GC信息
  • -Xms5M -Xmn5M 设置JVM内存大小是5M
  1. public static void main(String[] args){ 
  2.         for(int i = 0; i < 5_000_000; i++){ 
  3.             createObject(); 
  4.         } 
  5.     } 
  6.  
  7.     public static void createObject(){ 
  8.         new Object(); 
  9.     } 

运行结果是没有GC。

把虚拟机参数改成 -XX:+PrintGC -Xms5M -Xmn5M -XX:-DoEscapeAnalysis。关闭逃逸分析得到结果的部分截图是,说明了进行了GC,并且次数还不少。

  1. [GC (Allocation Failure)  4096K->504K(5632K), 0.0012864 secs] 
  2. [GC (Allocation Failure)  4600K->456K(5632K), 0.0008329 secs] 
  3. [GC (Allocation Failure)  4552K->424K(5632K), 0.0006392 secs] 
  4. [GC (Allocation Failure)  4520K->440K(5632K), 0.0007061 secs] 
  5. [GC (Allocation Failure)  4536K->456K(5632K), 0.0009787 secs] 
  6. [GC (Allocation Failure)  4552K->440K(5632K), 0.0007206 secs] 
  7. [GC (Allocation Failure)  4536K->520K(5632K), 0.0009295 secs] 
  8. [GC (Allocation Failure)  4616K->512K(4608K), 0.0005874 secs] 

这说明了JVM在逃逸分析之后,将对象分配在了方法createObject()方法栈上。方法栈上的对象在方法执行完之后,栈桢弹出,对象就会自动回收。这样的话就不需要等内存满时再触发内存回收。这样的好处是程序内存回收效率高,并且GC频率也会减少,程序的性能就提高了。

优化二 同步锁消除

如果发现某个对象只能从一个线程可访问,那么在这个对象上的操作可以不需要同步。

虚拟机配置参数:-XX:+PrintGC -Xms500M -Xmn500M -XX:+DoEscapeAnalysis。配置500M是保证不触发GC。

  1. public static void main(String[] args){ 
  2.         long start = System.currentTimeMillis(); 
  3.         for(int i = 0; i < 5_000_000; i++){ 
  4.             createObject(); 
  5.         } 
  6.         System.out.println("cost = " + (System.currentTimeMillis() - start) + "ms"); 
  7.     } 
  8.  
  9.     public static void createObject(){ 
  10.         synchronized (new Object()){ 
  11.  
  12.         } 
  13.     } 

运行结果

  1. cost = 6ms 

把逃逸分析关掉:-XX:+PrintGC -Xms500M -Xmn500M -XX:-DoEscapeAnalysis

运行结果

  1. cost = 270ms 

说明了逃逸分析把锁消除了,并在性能上得到了很大的提升。这里说明一下Java的逃逸分析是方法级别的,因为JIT的即时编译是方法级别。

优点三 分离对象或标量替换。

这个简单来说就是把对象分解成一个个基本类型,并且内存分配不再是分配在堆上,而是分配在栈上。这样的好处有,一、减少内存使用,因为不用生成对象头。 二、程序内存回收效率高,并且GC频率也会减少,总的来说和上面优点一的效果差不多。

OK,现在我们又知道了一件聪明的JVM在背后为我们做的事了。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/260196.html<

(0)
运维的头像运维
上一篇2025-05-03 01:26
下一篇 2025-05-03 01:27

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注