JVM詳解之匯編角度理解本地變量的生命周期
簡(jiǎn)介
java方法中定義的變量,它的生命周期是什么樣的呢?是不是一定要等到方法結(jié)束,這個(gè)創(chuàng)建的對(duì)象才會(huì)被回收呢?
帶著這個(gè)問(wèn)題我們來(lái)看一下今天的這篇文章。
本地變量的生命周期
在類(lèi)中,變量類(lèi)型有類(lèi)變量,成員變量和本地變量。
本地變量指的是定義在方法中的變量,如果我們?cè)诜椒ㄖ卸x了一個(gè)變量,那么這個(gè)變量的生命周期是怎么樣的呢?
舉個(gè)例子:
public void test(){ Object object = new Object(); doSomeThingElse(){ ... } }
在上面的test方法中,定義了一個(gè)object本地變量,然后又執(zhí)行了一個(gè)方法。
因?yàn)樵趈ava中,我們無(wú)法直接控制對(duì)象的生命周期,對(duì)象的回收是由垃圾回收器自動(dòng)進(jìn)行的。
通常來(lái)說(shuō)這個(gè)object對(duì)象會(huì)維持到整個(gè)test執(zhí)行結(jié)束才會(huì)被回收。
現(xiàn)在我們考慮一個(gè)特殊的情況,如果doSomeThingElse這個(gè)方法是一個(gè)while循環(huán),并且永遠(yuǎn)不會(huì)結(jié)束,那么這個(gè)創(chuàng)建出來(lái)的object對(duì)象會(huì)不會(huì)被回收呢?還是一直都存在內(nèi)存中?
先說(shuō)我們的結(jié)論,JVM非常智能,可以檢測(cè)出來(lái)這種情況,將object對(duì)象進(jìn)行回收。
舉例說(shuō)明
為了能夠更好的說(shuō)明問(wèn)題,我們自定義一個(gè)Test對(duì)象,并在其創(chuàng)建和被回收之前打印相應(yīng)的信息。
public static class Test { public Test() { System.out.println("創(chuàng)建對(duì)象 " + this); } public void test() { System.out.println("測(cè)試對(duì)象 " + this); } @Override protected void finalize() throws Throwable { System.out.println("回收對(duì)象 " + this); } }
然后做兩個(gè)測(cè)試,第一個(gè)測(cè)試沒(méi)有無(wú)限循環(huán),第二個(gè)測(cè)試保持無(wú)限循環(huán),循環(huán)通過(guò)一個(gè)volatile變量flag來(lái)控制:
public static void main(String[] args) throws InterruptedException { System.out.println("開(kāi)始測(cè)試1"); resetFlag(); flag = true; testLocalVariable(); System.out.println("等待Test1結(jié)束"); Thread.sleep(10000); System.out.println("開(kāi)始測(cè)試2"); flag = true; testLocalVariable(); }
看一下testLocalVariable方法的定義:
public static void testLocalVariable() { Test test1 = new Test(); Test test2 = new Test(); while (flag) { // 啥都不做 } test1.test(); }
然后我們?cè)賳?dòng)一個(gè)線程做定時(shí)的GC。好了一切就緒,我們運(yùn)行吧:
開(kāi)始測(cè)試1
創(chuàng)建對(duì)象 com.flydean.LocalVariableReachability$Test@119d7047
創(chuàng)建對(duì)象 com.flydean.LocalVariableReachability$Test@776ec8df
回收對(duì)象 com.flydean.LocalVariableReachability$Test@776ec8df
測(cè)試對(duì)象 com.flydean.LocalVariableReachability$Test@119d7047
等待Test1結(jié)束
回收對(duì)象 com.flydean.LocalVariableReachability$Test@119d7047開(kāi)始測(cè)試2
創(chuàng)建對(duì)象 com.flydean.LocalVariableReachability$Test@4eec7777
創(chuàng)建對(duì)象 com.flydean.LocalVariableReachability$Test@3b07d329
回收對(duì)象 com.flydean.LocalVariableReachability$Test@3b07d329
先看測(cè)試1的結(jié)果,我們可以看到第二個(gè)對(duì)象在調(diào)用test1.test()之前就被回收了。
再看測(cè)試2的結(jié)果,我們可以看到第二個(gè)對(duì)象同樣被回收了。
結(jié)果說(shuō)明了JVM是足夠智能的,可以自行優(yōu)化本地變量的生命周期。
優(yōu)化的原因
我們考慮一下,JVM是在什么階段對(duì)本地變量的生命周期進(jìn)行優(yōu)化的呢?
很明顯,這個(gè)優(yōu)化不是在編譯期間進(jìn)行的,而是在運(yùn)行期中進(jìn)行的優(yōu)化。
我們使用-XX:+PrintAssembly分析一下匯編代碼:
首先說(shuō)明,本人的匯編語(yǔ)言還是很多年前學(xué)過(guò)的,如果解釋起來(lái)有錯(cuò)誤的地方,請(qǐng)多多指正。
先說(shuō)兩個(gè)概念rbx和r10都是64位CPU的寄存器,r10d是r10的低32位。
先看紅框1, 紅框1表示rbx中保存的是我們定義的LocalVariableReachability類(lèi)中的一個(gè)Test對(duì)象。
再看紅框2,紅框2表示r10現(xiàn)在保存的是LocalVariableReachability這個(gè)類(lèi)實(shí)例。
紅框3表示的是進(jìn)入while循環(huán)的時(shí)候,ImutableOopMap中存儲(chǔ)的對(duì)象,大家可以看到里面只有r10和rbx,也就是說(shuō)只有類(lèi)實(shí)例和其中的一個(gè)Test實(shí)例。
紅框4是什么呢?紅框4表示的是一個(gè)safe point,也就是垃圾回收的時(shí)候的安全點(diǎn)。在這個(gè)安全點(diǎn)上如果有不再被使用的對(duì)象就會(huì)被回收。
因?yàn)镮mutableOopMap中只存有兩個(gè)對(duì)象,那么剩下的一個(gè)Test實(shí)例就會(huì)被回收。
到此這篇關(guān)于JVM詳解之匯編角度理解本地變量的生命周期的文章就介紹到這了,更多相關(guān)JVM匯編角度理解本地變量的生命周期內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java的常見(jiàn)熱門(mén)ORM框架優(yōu)缺點(diǎn)區(qū)別
Java?ORM框架是一種用于將Java對(duì)象映射到關(guān)系型數(shù)據(jù)庫(kù)中的工具,使得開(kāi)發(fā)人員能夠通過(guò)對(duì)象操作數(shù)據(jù)庫(kù)而不必直接使用SQL查詢,Java開(kāi)發(fā)變得更加高效和易于維護(hù),選擇適合你的ORM框架是根據(jù)你的需求決定的,比如你的應(yīng)用場(chǎng)景,數(shù)據(jù)結(jié)構(gòu)和技術(shù)水平等2024-02-02java計(jì)算給定字符串中出現(xiàn)次數(shù)最多的字母和該字母出現(xiàn)次數(shù)的方法
這篇文章主要介紹了java計(jì)算給定字符串中出現(xiàn)次數(shù)最多的字母和該字母出現(xiàn)次數(shù)的方法,涉及java字符串的遍歷、轉(zhuǎn)換及運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2017-02-02springboot短信驗(yàn)證碼登錄功能的實(shí)現(xiàn)
這篇文章主要介紹了springboot短信驗(yàn)證碼登錄功能的實(shí)現(xiàn),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02spring scheduled單線程和多線程使用過(guò)程中的大坑
本文主要介紹了spring scheduled單線程和多線程使用過(guò)程中的大坑,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01基于Java8并行流(parallelStream)的注意點(diǎn)
這篇文章主要介紹了Java8并行流(parallelStream)的注意點(diǎn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07springboot 通過(guò)代碼自動(dòng)生成pid的方法
這篇文章主要介紹了springboot 通過(guò)代碼自動(dòng)生成pid的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07詳解SpringCloudGateway內(nèi)存泄漏問(wèn)題
這篇文章主要介紹了詳解SpringCloudGateway內(nèi)存泄漏問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07