? ? ? ? 首先弄砍,我們要了解finally的實(shí)際中使用的場(chǎng)景——finally出自異常處理中的try{}catch{}finally{}語(yǔ)句塊中,其基本的目的是確保無(wú)論異常是否發(fā)生钧栖,都必然會(huì)執(zhí)行finally語(yǔ)句塊中的內(nèi)容。
? ? ? ? 下面,我來(lái)講講fianlly語(yǔ)句塊的切實(shí)應(yīng)用芯急。
一失驶、程序因代碼或非個(gè)人因素終止時(shí)finally代碼不執(zhí)行土居。
? ? ? ? 這里的非個(gè)人因素是指遇到設(shè)備關(guān)機(jī)或硬件設(shè)備損壞等原因?qū)е鲁绦驘o(wú)法再正常執(zhí)行,這是不可控的,不多做關(guān)注擦耀。
? ? ? ? 因代碼原因終止的情況有很多棉圈,有:調(diào)用System.exit()函數(shù),調(diào)用halt()函數(shù)眷蜓,try代碼中因循環(huán)溢出導(dǎo)致系統(tǒng)崩潰和守護(hù)線程的原因分瘾。下面,我會(huì)對(duì)這些方法進(jìn)行一些展開吁系。
Ⅰ德召、調(diào)用System.exit()函數(shù)的方法
? ? ? ? 可以從API上直觀的看到,該方法用于終止虛擬機(jī)汽纤,而當(dāng)虛擬機(jī)被終止了上岗,程序運(yùn)行也就結(jié)束了。
? ? ? ? 下面時(shí)簡(jiǎn)單的代碼演示:
? ? ? ? 可以看到finally語(yǔ)句塊中的輸出finally語(yǔ)句塊被執(zhí)行沒(méi)有執(zhí)行蕴坪,而當(dāng)我們注釋掉語(yǔ)句Ststem.exit(1);語(yǔ)句后肴掷,finally中的語(yǔ)句塊被執(zhí)行了(后續(xù)不會(huì)再進(jìn)行注釋操作男翰,第一次做注釋操作便于前后做參照俊马,表明確實(shí)是Ststem.exit(1);語(yǔ)句生效導(dǎo)致finally語(yǔ)句塊不輸出)。
Ⅱ慨蛙、調(diào)用halt()函數(shù)的方法
? ? ? ? 同樣我們可以從API上找到相關(guān)方法续室,它隸屬于Runtime類栋烤。可以看到它的使用與Ststem.exit()方法的使用是一樣的挺狰,都是強(qiáng)制終止當(dāng)前運(yùn)行Java虛擬機(jī)明郭。
? ? ? ? 下面做代碼演示:
? ? ? ? 作用與使用方法上看,Ststem.exit()與Runtime.getRuntimr.halt()方法沒(méi)有太大差別(區(qū)別涉及線程)丰泊。
Ⅲ薯定、try語(yǔ)句塊循環(huán)溢出
? ? ? ? 正如字面意思,try語(yǔ)句塊陷入死循環(huán)導(dǎo)致虛擬機(jī)終止程序?qū)⒉粫?huì)執(zhí)行finally語(yǔ)句塊中的內(nèi)容瞳购。
? ? ? ? 下面做代碼演示:
Ⅳ话侄、守護(hù)線程
? ? ? ? 如果守護(hù)線程剛開始執(zhí)行到finally代碼塊,此時(shí)沒(méi)有任何其他非守護(hù)線程学赛,那么虛擬機(jī)將退出年堆,此時(shí) JVM 不會(huì)等待守護(hù)線程的 finally 代碼塊執(zhí)行完成。(涉及線程【線程中守護(hù)線程生命周期與主線程一起一樣盏浇,主線程結(jié)束变丧,守護(hù)線程即便有未執(zhí)行的finally依然會(huì)強(qiáng)制結(jié)束】)
二、finally代碼執(zhí)行時(shí)遇到的問(wèn)題绢掰。
? ? ? ? 使用finally語(yǔ)句塊時(shí)遇到的問(wèn)題基本可以分為兩類痒蓬,一類是上述的不執(zhí)行問(wèn)題童擎,另一類則是執(zhí)行時(shí)finally語(yǔ)句塊與return返回值之間的問(wèn)題(不涉及返回值與強(qiáng)制終止的finally語(yǔ)句塊將必定被執(zhí)行,必定被執(zhí)行攻晒,必定被執(zhí)行)顾复。
? ? ? ? 而finally語(yǔ)句塊與return之間的情景又大致分為以下兩種情況:try或catch語(yǔ)句存在返回值,而finally語(yǔ)句塊有返回值或拋出異常語(yǔ)句鲁捏;try或catch語(yǔ)句存在返回值芯砸,而finally語(yǔ)句塊無(wú)返回值但對(duì)返回的值進(jìn)行修改。我將對(duì)其做簡(jiǎn)單的分析给梅。
Ⅰ乙嘀、finally語(yǔ)句塊有返回值或拋出異常語(yǔ)句
? ? ? ? 無(wú)視前面try和catch語(yǔ)句中的返回值,執(zhí)行finally中的返回值或?qū)惓伋觥?/p>
Ⅱ破喻、finally語(yǔ)句塊無(wú)返回值但對(duì)返回的值進(jìn)行修改
? ? ? ? 首先,我們要對(duì)return和finally執(zhí)行次序進(jìn)行探討盟榴。毫無(wú)疑問(wèn)曹质,try中的return先執(zhí)行,但他并沒(méi)有直接將值返回給函數(shù)擎场,而是將值備份后羽德,再執(zhí)行完finally語(yǔ)句塊才會(huì)返回給函數(shù)。而在值備份后到程序結(jié)束前的這段時(shí)間里迅办,finally語(yǔ)句塊可能會(huì)對(duì)返回的值造成影響宅静。
? ? ? ? 這又分作兩種情況:1.返回值是引用類型;2.返回值是數(shù)據(jù)類型(返回值無(wú)影響但與數(shù)據(jù)類型做區(qū)分)站欺。下面我會(huì)對(duì)上面兩種情況進(jìn)行展開姨夹。
1.返回值是引用類型
? ? ? ? 當(dāng)返回值是引用類型時(shí),return備份的值是引用類型對(duì)象指向的地址矾策,finally語(yǔ)句塊對(duì)其對(duì)象進(jìn)行操作時(shí)會(huì)改變指向的地址中內(nèi)存的值磷账,從而使得結(jié)果返回值與只執(zhí)行try語(yǔ)句時(shí)的返回值不太一樣。
? ? ? ? 下面做代碼演示:
? ? ? ? 可以看到返回的對(duì)象t中存放的a的值并不是try中return方法備份的贾虽,t對(duì)象中a的值可以被finally方法修改逃糟。
2.返回值是數(shù)據(jù)類型
? ? ? ? try語(yǔ)句塊中return備份的值即為那個(gè)數(shù)據(jù)的具體數(shù)值,finally語(yǔ)句對(duì)其進(jìn)行改變時(shí)不會(huì)造成影響蓬豁,這里獨(dú)立列出來(lái)主要是和引用數(shù)據(jù)區(qū)分绰咽。
? ? ? ? 下面做代碼演示:
? ? ? ? 可以看到雖然finally語(yǔ)句塊中對(duì)返回值a進(jìn)行了操作,但必不能影響返回值地粪。
結(jié)語(yǔ)
本人第一次做技術(shù)博客取募,不足之處望批評(píng)指正。
參考文章:
https://juejin.im/post/5e1346ba5188253a6b371d78?utm_source=gold_browser_extension#heading-12
工具:JDK11API