摘要:記得幾年前有一次棧長去面試,問到了這么一個問題中的對象都是在堆中分配嗎說明為什么當時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當時我壓根就不知道他在考什么知識點,難道對象不是在堆中分配嗎最后就沒然后了,回去等通知了。。
記得幾年前有一次棧長去面試,問到了這么一個問題:
Java中的對象都是在堆中分配嗎?說明為什么!
當時我被問得一臉蒙逼,瞬間被秒殺得體無完膚,當時我壓根就不知道他在考什么知識點,難道對象不是在堆中分配嗎?最后就沒然后了,回去等通知了。。
這個面試題很經典,我最近也分享到了知識星球上面:
回答很精彩,大家可以加入一起搞技術,我現在將答案總結一下給大家。
什么是逃逸分析?關于 Java 逃逸分析的定義:
逃逸分析(Escape Analysis)簡單來講就是,Java Hotspot 虛擬機可以分析新創建對象的使用范圍,并決定是否在 Java 堆上分配內存的一項技術。
逃逸分析的 JVM 參數如下:
開啟逃逸分析:-XX:+DoEscapeAnalysis
關閉逃逸分析:-XX:-DoEscapeAnalysis
顯示分析結果:-XX:+PrintEscapeAnalysis
逃逸分析技術在 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、全局逃逸(GlobalEscape)即一個對象的作用范圍逃出了當前方法或者當前線程,有以下幾種場景:
對象是一個靜態變量
對象是一個已經發生逃逸的對象
對象作為當前方法的返回值
2、參數逃逸(ArgEscape)即一個對象被作為方法參數傳遞或者被參數引用,但在調用過程中不會發生全局逃逸,這個狀態是通過被調方法的字節碼確定的。
3、沒有逃逸即方法中的對象沒有發生逃逸。
逃逸分析優化針對上面第三點,當一個對象沒有逃逸時,可以得到以下幾個虛擬機的優化。
1) 鎖消除
我們知道線程同步鎖是非常犧牲性能的,當編譯器確定當前對象只有當前線程使用,那么就會移除該對象的同步鎖。
例如,StringBuffer 和 Vector 都是用 synchronized 修飾線程安全的,但大部分情況下,它們都只是在當前線程中用到,這樣編譯器就會優化移除掉這些鎖操作。
鎖消除的 JVM 參數如下:
開啟鎖消除:-XX:+EliminateLocks
關閉鎖消除:-XX:-EliminateLocks
鎖消除在 JDK8 中都是默認開啟的,并且鎖消除都要建立在逃逸分析的基礎上。
2) 標量替換
首先要明白標量和聚合量,基礎類型和對象的引用可以理解為標量,它們不能被進一步分解。而能被進一步分解的量就是聚合量,比如:對象。
對象是聚合量,它又可以被進一步分解成標量,將其成員變量分解為分散的變量,這就叫做標量替換。
這樣,如果一個對象沒有發生逃逸,那壓根就不用創建它,只會在?;蛘呒拇嫫魃蟿摻ㄋ玫降某蓡T標量,節省了內存空間,也提升了應用程序性能。
標量替換的 JVM 參數如下:
開啟標量替換:-XX:+EliminateAllocations
關閉標量替換:-XX:-EliminateAllocations
顯示標量替換詳情:-XX:+PrintEliminateAllocations
標量替換同樣在 JDK8 中都是默認開啟的,并且都要建立在逃逸分析的基礎上。
3) 棧上分配
當對象沒有發生逃逸時,該對象就可以通過標量替換分解成成員標量分配在棧內存中,和方法的生命周期一致,隨著棧幀出棧時銷毀,減少了 GC 壓力,提高了應用程序性能。
總結逃逸分析講完了,總結了不少時間,我們也應該大概知道逃逸分析是為了優化 JVM 內存和提升程序性能的。
我們知道這點后,在平時開發過程中就要可盡可能的控制變量的作用范圍了,變量范圍越小越好,讓虛擬機盡可能有優化的空間。
簡單舉一個例子吧,如:
return sb;
可以改為:
return sb.toString();
這是一種優化案例,把 StringBuilder 變量控制在了當前方法之內,沒有逃出當前方法作用域。
大家還有沒有別的優化經驗,歡迎分享~
參考資料:
https://docs.oracle.com/javas...
https://blog.csdn.net/rickiye...
https://blog.csdn.net/baichou...
關注Java技術棧微信公眾號,棧長將繼續分享 Java 干貨教程,公眾號第一時間推送,持續關注。在公眾號后臺回復:java,獲取棧長整理的更多的 Java 教程,都是實戰干貨,以下僅為部分預覽。
你真的搞懂 transient 關鍵字了嗎?
面試??迹篠ynchronized 有幾種用法?
Java 11 已發布,String 還能這樣玩!
Java 中的 String 真的是不可變嗎?
sleep( ) 和 wait( ) 的這 5 個區別
……
本文原創首發于微信公眾號:Java技術棧(id:javastack),轉載請原樣保留本信息。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/77742.html
摘要:我自己總結的學習的系統知識點以及面試問題,已經開源,目前已經。面試官那你都了解里面的哪些東西呢我哈哈哈這可是我的強項,從,說到,,又說到線程池,分別說了底層實現和項目中的應用。 我自己總結的Java學習的系統知識點以及面試問題,已經開源,目前已經 35k+ Star。會一直完善下去,歡迎建議和指導,同時也歡迎Star: https://github.com/Snailclimb... ...
摘要:那在屆有哪些被封神的庫呢,今天就給大家介紹十個堪稱殺手級別的工具包。總之,是一款非常強大且易用的網絡請求庫。地址最好用的框架。圖像處理,該庫是你的不二之選。 Pyt...
摘要:那在屆有哪些被封神的庫呢,今天就給大家介紹十個堪稱殺手級別的工具包。該庫是在之上完善的,自此請求就變得異常的簡單,一行代碼即可搞定。這是最流行的網絡爬蟲框架庫,沒有之一。最好用的框架。圖像處理,該庫是你的不二之選。 文 |?豆豆 來源:Python 技術「ID: pythonall」 ...
摘要:序列化對象要保留充分的信息,用來恢復數據對象,但是為了節約存儲空間和網絡帶寬,序列化出的二進制流要盡可能小。序列化序列化是一種支持動態類型跨語言基于對象傳輸的網絡協議,對象序列化后的二進制流,可以被其他語言反序列化。 記得很久以前寫代碼的時候,每次新建一個實體都會下意識的繼承Serializable接口,大部分人都知道這是對對象的序列化,可是你們真的知道序列化嗎?這篇文章就簡單的說下j...
閱讀 1487·2021-11-24 11:16
閱讀 2689·2021-07-28 12:32
閱讀 2302·2019-08-30 11:22
閱讀 1440·2019-08-30 11:01
閱讀 595·2019-08-29 16:24
閱讀 3547·2019-08-29 12:52
閱讀 1625·2019-08-29 12:15
閱讀 1332·2019-08-29 11:18