avatar

目录
JVM内存逃逸
  • 第一次听到JVM内存逃逸的名词时还是很懵逼的,于是赶紧各种查资料,终于搞懂了这个地方。

  • JVM的内存分配主要在是运行时数据区(Runtime Data Areas),而运行时数据区又分为了:方法区堆区PC寄存器Java虚拟机栈(就是栈区,官方文档还是叫Java虚拟机栈)本地方法区,而内存逃逸主要是对象的动态作用域的改变而引起的,故而内存逃逸的分析就是分析对象的动态作用域。

  • 方法逃逸

    • 什么是方法逃逸呢?举个栗子,在一个方法中定义一个对象后(方法内局部的),这个对象被外部方法引用,比如作为返回值返回传递到其他的地方,当这个方法执行结束要进行GC时,这个方法中的对象本应该被回收,却发现该对象还是存活状态没法回收,就称为方法逃逸

    • 上代码:

      java
      1
      2
      3
      4
      5
      6
      public static StringBuffer getStringBuffer(String str1,String str2) {
      StringBuffer stringBuffer = new StringBuffer();
      stringBuffer.append(str1);
      stringBuffer.append(str2);
      return stringBuffer;
      }

      上面的代码中的stringBuffer虽然是方法内的局部变量,因为stringBuffer被当作返回值返回,这样stringBuffer可能被其他的方法所改变,作用域就不仅仅在本方法内啦,这样就是逃逸到了方法外部。对的,就是越狱了。

    • 怎么样才能不让stringBuffer逃出方法呢?那么不直接返回stringBuffer对象不就可以了嘛!如下面的代码:

      java
      1
      2
      3
      4
      5
      6
      public static StringBuffer getStringBuffer(String str1,String str2) {
      StringBuffer stringBuffer = new StringBuffer();
      stringBuffer.append(str1);
      stringBuffer.append(str2);
      return stringBuffer.toString();
      }
  • 线程逃逸:上面的例子,直接将对象返回,该对象可能被外部线程访问,如:赋值给类变量等,称为线程逃逸。

  • 总的来说就是一个对象的指针被多个方法或者线程引用时,我们就称这个对象的指针发生了逃逸。

  • 优化:即证明一个对象不会逃逸到方法或线程外。

    • 栈上分配

      说起对象,那你第一个想到的是在堆空间上进行内存分配,GC在堆空间上筛选可回收的对象,回收对象,整理内存都需要浪费时间,若能通过逃逸分析确定某些对象是一定不会逃逸出方法之外的,就可以直接让这个对象在栈上分配内存,该对象随方法的执行结束栈帧出栈而销毁,减轻了GC的压力。

    • 同步消除

      线程同步本身比较耗时,若确定了一个变量不会逃逸出线程,无法被其他线程访问到,那这个变量的读写就不会存在竞争,这个变量的同步措施就可以清除掉。

    • 标量替换

      标量:Java中的原始数据类型(int,char,long等)都不能再进一步分解,他们就可以称为标量。

      聚合量:若一个数据可以继续分解,那就称之为聚合量,而对象就是典型的聚合量。

      若逃逸分析证明一个对象不会逃逸出方法,不会被外部访问,并且这个对象是可以被分解的,那程序在真正执行的时候可能不创建这个对象,而是直接创建这个对象分解后的标量来代替。这样就无需在对对象分配空间了,只在栈上为分解出的变量分配内存即可。

  • All in all

    • 逃逸分析是比较耗时的,所以性能未必提升很多,因为其耗时性,采用的算法都是不那么准确但是时间压力相对较小的算法来完成的,这就可能导致效果不稳定,要慎用。
    • 由于HotSpot虚拟机目前的实现方法导致栈上分配实现起来比较复杂,所以HotSpot虚拟机中暂时还没有这项优化。
    • 相关的JVM参数
      • -XX:+DoEscapeAnalysis 开启逃逸分析、
      • -XX:+PrintEscapeAnalysis 开启逃逸分析后,可通过此参数查看分析结果。
      • -XX:+EliminateAllocations 开启标量替换
      • -XX:+EliminateLocks 开启同步消除
      • -XX:+PrintEliminateAllocations 开启标量替换后,查看标量替换情况。
文章作者: XiaoMing
文章链接: https://xiaoming0929.github.io/2020/04/02/JVM%E5%86%85%E5%AD%98%E9%80%83%E9%80%B8/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 XiaoMing's Blog
打赏
  • 微信
    微信
  • 支付寶
    支付寶