成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

JVM之逃逸分析

云計算 虛擬化
在編譯程序優化理論中,逃逸分析是一種確定指針動態范圍的方法——分析在程序的哪些地方可以訪問到指針。它涉及到指針分析和形狀分析。

[[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在背后為我們做的事了。

責任編輯:武曉燕 來源: 今日頭條
相關推薦

2024-04-07 11:33:02

Go逃逸分析

2017-01-12 14:52:03

JVMFinalRefere源碼

2021-10-14 10:22:19

逃逸JVM性能

2018-07-09 15:11:14

Java逃逸JVM

2023-01-10 09:18:37

Go內存分配逃逸

2017-01-11 14:02:32

JVM源碼內存

2024-12-09 09:50:00

JVM逃逸逃逸分析

2017-02-27 11:48:58

JVM源碼分析Java

2020-05-26 18:50:46

JVMAttachJava

2024-01-17 08:02:26

Java逃逸分配

2022-07-10 23:15:46

Go語言內存

2024-12-17 07:41:34

Java逃逸分析

2023-01-28 08:32:04

Go內存分配

2012-01-11 11:28:00

JavaJVM

2010-09-26 16:55:31

JVM學習筆記

2022-11-30 08:19:15

內存分配Go逃逸分析

2017-01-11 14:19:26

JVM源碼All

2024-05-13 08:05:26

JVMJava逃逸分析

2019-09-16 09:46:55

對抗反分析檢測逃逸惡意軟件

2019-09-16 09:46:55

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产99久久精品一区二区永久免费 | 久久99精品视频 | 国产a视频 | 国产在线不卡 | 欧美mv日韩mv国产网站91进入 | 国产亚洲二区 | 欧美精品一区二区三区在线播放 | 久久机热| 在线欧美亚洲 | 成人精品在线观看 | 国产99久久精品一区二区永久免费 | 欧美群妇大交群中文字幕 | 黄色毛片免费 | 天天射中文 | 国产成人av一区二区三区 | 欧美精品一区二区在线观看 | 欧美黄色网 | 亚洲精品欧美一区二区三区 | 久久久久久久97 | 亚洲欧美精品在线观看 | 一本一道久久a久久精品蜜桃 | 人人叉| 亚洲免费一区二区 | 久久久国产精品入口麻豆 | 看av片网站| 亚洲国产精品久久久久久 | 国产成人a亚洲精品 | 欧美亚洲另类丝袜综合网动图 | 中文字幕在线一 | 国产乱码精品一区二区三区忘忧草 | 午夜成人免费视频 | 91xxx在线观看 | 日本久久久久久 | 精品影院 | 亚洲精品日日夜夜 | 中文精品一区二区 | 天堂亚洲网 | 久久不卡 | 国产精品不卡 | 国产欧美精品在线观看 | 一级毛片视频在线 |