3分鐘搞清楚 JVM逃逸分析
作為一個合格java開發者都知道,基本上所有物件都是在堆上創建。但是,這裡還是沒有把話說絕對哈,指的是基本上所有。
昨天一位朋友在面試中,就說了所有物件都在堆中創建,然後背面試官一陣的嘲笑。
開始我們的正文,我們今天來聊聊關於逃逸分析。
逃逸分析(Escape Analysis)是目前Java虛擬機器中較前沿的最佳化技術。這是一種可以有效減少Java 程式中同步負載和記憶體堆分配壓力的跨函數全域資料流分析演算法。透過逃逸分析,Java Hotspot編譯器能夠分析出一個新的物件的引用的使用範圍從而決定是否要將這個物件分配到堆上。
逃逸分析的基本原理是:分析物件動態作用域,當一個物件在方法裡面被定義後,它可能被外部方法所引用,例如作為呼叫參數傳遞到其他方法中,這種稱為方法逃逸;甚至還有可能被外部執行緒存取到,譬如賦值給可以在其他執行緒中存取的實例變量,這種稱為執行緒逃逸;從不逃逸、方法逃逸到執行緒逃逸,稱為物件由低到高的不同逃逸程度。
開啟逃逸分析,編譯器可以對程式碼進行如下最佳化:
#同步消除:如果一個物件被逃逸分析發現只能被一個執行緒所訪問,那對於這個物件的操作可以不同步。 堆疊上分配:如果確定物件不會逃逸出執行緒之外,那麼讓這個物件在堆疊上分配記憶體將會是一個很不錯的主意,而物件所佔用的記憶體空間就可以隨堆疊幀出棧而銷毀。 標量替換:如果一個物件被逃逸分析發現不會被外部方法訪問,並且這個物件可以拆散,那麼程式真正執行的時候將可能不去建立這個對象,而改為直接創建它的若干個比這個方法使用的成員變數來代替。 將物件拆分後,可以讓物件的成員變數在堆疊上分配和讀寫。
JVM中透過以下參數可以指定是否開啟逃逸分析:
##-XX: DoEscapeAnalysis :表示開啟逃逸分析(JDK 1.7之後預設開啟)。
-XX:-DoEscapeAnalysis :表示關閉逃脫分析。
同步消除
線程同步本身是一個相對耗時的過程,如果逃逸分析能夠確定一個變數不會逃逸出線程,無法被其他線程訪問,那麼這個變數的讀寫肯定就不會有競爭,對這個變數實施的同步措施也就可以安全地消除掉。
如以下程式碼:
public void method() { Object o = new Object(); synchronized (o) { System.out.println(o); } }
對物件o
加鎖,但是物件o的生命週期與方法method()一樣,所以不會被其他執行緒存取到,不會發生執行緒安全性問題,那麼在JIT編譯階段會被最佳化為如下所示:
public void method() { Object o = new Object(); System.out.println(o); }
這也稱為鎖定消除。
堆疊上分配
在Java虛擬機器中,Java堆疊上分配建立物件的記憶體空間幾乎是Java程式設計師都知道的常識,Java堆中的物件對於各個執行緒都是共享且可見的,只要持有這個物件的引用,就可以存取到堆中儲存的物件資料。虛擬機器的垃圾收集子系統會回收堆中不再使用的對象,但回收動作無論是標記篩選出可回收對象,還是回收和整理內存,都需要耗費大量資源。但是,存在一種特殊情況,如果逃逸分析確認物件不會逃逸出執行緒之外,那麼就可能被最佳化成堆疊上分配。這樣就無需在堆上分配內存,也無須進行垃圾回收了。
如下列程式碼:
public static void main(String[] args) throws InterruptedException { for (int i = 0; i < 1000000; i++) { alloc(); } Thread.sleep(100000); } private static void alloc() { User user = new User(); }
程式碼很簡單,就是循環建立100萬次,使用alloc()方法建立100萬個User物件。這裡的alloc()方法中定義了User物件並沒有被其他方法引用,所以符合堆疊上分配的要求。
JVM參數如下:
-Xmx2G -Xms2G -XX:+DoEscapeAnalysis -XX:+PrintGCDetails -XX:+HeapDumpOnOutOfMemoryError
啟動程序,透過jmap工具檢視實例數:
jmap -histo pid num #instances #bytes class name ---------------------------------------------- 1: 3771 2198552 [B 2: 10617 1722664 [C 3: 104057 1664912 com.miracle.current.lock.StackAllocationTest$User
我們可以看到程式總共建立了104057個User對象,遠小於100萬。我們可以關閉逃逸分析再來看下:
-Xmx2G -Xms2G -XX:-DoEscapeAnalysis -XX:+PrintGCDetails -XX:+HeapDumpOnOutOfMemoryError
啟動程序,透過jmap工具查看實例數:
jmap -histo 42928 num #instances #bytes class name ---------------------------------------------- 1: 628 22299176 [I 2: 1000000 16000000 com.miracle.current.lock.StackAllocationTest$User
可以看到,關閉逃逸分析後總共創建了100萬個User物件。比較來看,堆疊上分配對堆疊記憶體消耗,GC都有著重要的作用。
標量替換
如果一個資料已經無法再分解成更小的資料來表示了,Java虛擬機器中的原始資料類型(int 、long 等數值型別及reference型別等)都不能再進一步分解了,那麼這些資料就可以稱為標量。相對的,如果一個資料可以繼續分解,那麼它就被稱為聚合量(Aggregate),Java中的物件就是典型的聚合量。
假如逃逸分析能夠證明一個對像不會被方法外部訪問,並且這個對象可以被拆散,那麼程序真正執行的時候將可能不去創建這個對象,而改為直接創建它的若干個被這個方法使用的成員變數來代替。
有以下程式碼:
public static void main(String[] args) { method(); } private static void method() { User user = new User(25); System.out.println(user.age); } private static class User { private int age; public User(int age) { this.age = age; } }
在method()
方法中创建User对象,指定age为25,这里User不会被其他方法引用,也就是说它不会逃逸出方法,并且User是可以拆解为标量的。所以alloc()
代码会优化为如下:
private static void alloc() { int age = 25; System.out.println(age); }
总结
尽管目前逃逸分析技术仍在发展之中,未完全成熟,但它是即时编译器优化技术的一个重要前进方向,在日后的Java虚拟机中,逃逸分析技术肯定会支撑起一系列更实用、有效的优化技术。
以上是3分鐘搞清楚 JVM逃逸分析的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

掌握JVM記憶體使用量的重點與注意事項JVM(JavaVirtualMachine)是Java應用程式運作的環境,其中最為重要的就是JVM的記憶體管理。合理地管理JVM記憶體不僅可以提高應用程式的效能,還可以避免記憶體洩漏和記憶體溢位等問題。本文將介紹JVM記憶體使用的要點和注意事項,並提供一些具體的程式碼範例。 JVM記憶體分區JVM記憶體主要分為以下區域:堆(He

該專案為了方便開發者更快監控多個遠端主機jvm,如果你的專案是Spring boot那麼很方便集成,jar包引入即可,不是Spring boot也不用氣餒,你可以快速自行初始化一個Spirng boot程式引入jar包即可

透過JVM命令列參數,您可以細微地調整JVM行為。其中通用參數包括:設定Java堆大小(-Xms、-Xmx)設定新生代大小(-Xmn)啟用平行垃圾收集器(-XX:+UseParallelGC)減少Survivor區記憶體佔用(-XX:-ReduceSurvivorSetInMemory)消除冗餘餘垃圾回收(-XX:-EliminateRedundantGCs)列印垃圾回收資訊(-XX:+PrintGC)使用G1垃圾收集器(-XX:-UseG1GC)設定最大垃圾回收暫停時間(-XX:MaxGCPau

Java是一種流行的程式語言,在開發Java應用程式的過程中,可能會遇到JVM記憶體溢位錯誤。這種錯誤通常會導致應用程式崩潰,影響用戶體驗。本文將探討JVM記憶體溢位錯誤的原因和如何處理和避免這種錯誤。 JVM記憶體溢位錯誤是什麼? Java虛擬機器(JVM)是Java應用程式的運作環境。在JVM中,記憶體被分為多個區域,其中包括堆疊、方法區、堆疊等。堆是用於存儲創建的對象的

JVM虛擬機的作用及原理解析簡介:JVM(JavaVirtualMachine)虛擬機是Java程式語言的核心組成部分之一,它是Java的最大賣點之一。 JVM的作用是將Java原始碼編譯成字節碼,並負責執行這些字節碼。本文將介紹JVM的作用及其運作原理,並提供一些程式碼範例以幫助讀者更好地理解。作用:JVM的主要作用是解決了不同平台上Java程式的可移

JVM原理詳解:深入探究Java虛擬機的工作原理,需要具體程式碼範例一、引言隨著Java程式語言的快速發展和廣泛應用,Java虛擬機(JavaVirtualMachine,簡稱JVM)也成為了軟體開發中不可或缺的一部分。 JVM作為Java程式的運作環境,能夠提供跨平台的特性,使得Java程式能夠在不同的作業系統上運作。在本文中,我們將深入探討JVM的工作原

JVM記憶體參數設定:如何合理調整堆記憶體大小?在Java應用程式中,JVM是負責管理記憶體的關鍵元件。其中,堆記憶體是用來儲存物件實例的地方,堆記憶體的大小設定對應用程式的效能和穩定性有著重要影響。本文將介紹如何合理調整堆記憶體大小的方法,並附帶具體程式碼範例。首先,我們需要了解一些關於JVM記憶體的基礎知識。 JVM的記憶體分成了幾個區域,包括堆疊記憶體、堆疊記憶體、方法區等。其中

在寫java程式來檢查JVM是32位元還是64位元之前,我們先討論一下JVM。 JVM是java虛擬機,負責執行字節碼。它是Java執行時間環境(JRE)的一部分。我們都知道java是平台無關的,但是JVM是平台相關的。我們需要為每個作業系統提供單獨的JVM。如果我們有任何java原始碼的字節碼,由於JVM,我們可以輕鬆地在任何平台上運行它。 java檔案執行的整個過程如下-首先,我們保存擴展名為.java的java原始碼,編譯器將其轉換為擴展名為.class的字節碼。這發生在編譯時。現在,在運行時,J
