記得幾年前有一次棧長去面試,問到了這麼一個問題:html
Java中的對象都是在堆中分配嗎?說明爲何!java
當時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當時我壓根就不知道他在考什麼知識點,難道對象不是在堆中分配嗎?最後就沒而後了,回去等通知了。。面試
這個面試題很經典,我最近也分享到了知識星球上面:算法
回答很精彩,你們能夠加入一塊兒搞技術,我如今將答案總結一下給你們。安全
關於 Java 逃逸分析的定義:微信
逃逸分析(Escape Analysis)簡單來說就是,Java Hotspot 虛擬機能夠分析新建立對象的使用範圍,並決定是否在 Java 堆上分配內存的一項技術。oracle
逃逸分析的 JVM 參數以下:ide
逃逸分析技術在 Java SE 6u23+ 開始支持,並默認設置爲啓用狀態,能夠不用額外加這個參數。性能
Java Hotspot 編譯器實現下面論文中描述的逃逸算法:優化
[Choi99] Jong-Deok Choi, Manish Gupta, Mauricio Seffano, Vugranam C. Sreedhar, Sam Midkiff, "Escape Analysis for Java", Procedings of ACM SIGPLAN OOPSLA Conference, November 1, 1999
根據 Jong-Deok Choi, Manish Gupta, Mauricio Seffano,Vugranam C. Sreedhar, Sam Midkiff 等大牛在論文《Escape Analysis for Java》中描述的算法進行逃逸分析的。
該算法引入了連通圖,用連通圖來構建對象和對象引用之間的可達性關係,並在次基礎上,提出一種組合數據流分析法。
因爲算法是上下文相關和流敏感的,而且模擬了對象任意層次的嵌套關係,因此分析精度較高,只是運行時間和內存消耗相對較大。
咱們瞭解了 Java 中的逃逸分析技術,再來了解下一個對象的逃逸狀態。
即一個對象的做用範圍逃出了當前方法或者當前線程,有如下幾種場景:
即一個對象被做爲方法參數傳遞或者被參數引用,但在調用過程當中不會發生全局逃逸,這個狀態是經過被調方法的字節碼肯定的。
即方法中的對象沒有發生逃逸。
針對上面第三點,當一個對象沒有逃逸時,能夠獲得如下幾個虛擬機的優化。
1) 鎖消除
咱們知道線程同步鎖是很是犧牲性能的,當編譯器肯定當前對象只有當前線程使用,那麼就會移除該對象的同步鎖。
例如,StringBuffer 和 Vector 都是用 synchronized 修飾線程安全的,但大部分狀況下,它們都只是在當前線程中用到,這樣編譯器就會優化移除掉這些鎖操做。
鎖消除的 JVM 參數以下:
鎖消除在 JDK8 中都是默認開啓的,而且鎖消除都要創建在逃逸分析的基礎上。
2) 標量替換
首先要明白標量和聚合量,基礎類型和對象的引用能夠理解爲標量,它們不能被進一步分解。而能被進一步分解的量就是聚合量,好比:對象。
對象是聚合量,它又能夠被進一步分解成標量,將其成員變量分解爲分散的變量,這就叫作標量替換。
這樣,若是一個對象沒有發生逃逸,那壓根就不用建立它,只會在棧或者寄存器上建立它用到的成員標量,節省了內存空間,也提高了應用程序性能。
標量替換的 JVM 參數以下:
標量替換一樣在 JDK8 中都是默認開啓的,而且都要創建在逃逸分析的基礎上。
3) 棧上分配
當對象沒有發生逃逸時,該對象就能夠經過標量替換分解成成員標量分配在棧內存中,和方法的生命週期一致,隨着棧幀出棧時銷燬,減小了 GC 壓力,提升了應用程序性能。
逃逸分析講完了,總結了很多時間,咱們也應該大概知道逃逸分析是爲了優化 JVM 內存和提高程序性能的。
咱們知道這點後,在平時開發過程當中就要可儘量的控制變量的做用範圍了,變量範圍越小越好,讓虛擬機儘量有優化的空間。
簡單舉一個例子吧,如:
return sb;
能夠改成:
return sb.toString();
這是一種優化案例,把 StringBuilder 變量控制在了當前方法以內,沒有逃出當前方法做用域。
你們還有沒有別的優化經驗,歡迎分享~
參考資料:
關注Java技術棧微信公衆號,棧長將繼續分享 Java 乾貨教程,公衆號第一時間推送,持續關注。在公衆號後臺回覆:java,獲取棧長整理的更多的 Java 教程,都是實戰乾貨,如下僅爲部分預覽。
本文原創首發於微信公衆號:Java技術棧(id:javastack),轉載請原樣保留本信息。