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