溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

詳解IDisposable和Finalize的意義

發(fā)布時(shí)間:2020-06-25 21:10:47 來(lái)源:網(wǎng)絡(luò) 閱讀:1567 作者:cnn237111 欄目:編程語(yǔ)言

本文主要討論IDisposable和Finalize如何釋放資源,以及微軟提供并推薦的一個(gè)標(biāo)準(zhǔn)IDisposable編程模式。

本文寫的通俗易懂,如果你還看不懂,就自己去反省一下為什么自己水平這么爛....


IDisposable接口主要提供一個(gè)Dispose方法,該方法用來(lái)釋放資源,準(zhǔn)確的說(shuō)是用來(lái)釋放非托管資源。理論上不是必須有 IDisposable 接口,你也能自己設(shè)計(jì)出一個(gè)釋放非托管資源的方法,但是使用這個(gè)接口的話,大家都統(tǒng)一使用,比較好理解,并且使用這個(gè)接口的類,可以使用using這種語(yǔ)法糖。

Finalize其實(shí)就是析構(gòu)函數(shù),學(xué)過(guò)C++的都知道啥叫析構(gòu)函數(shù),C#也一樣。你必須顯式的定義一個(gè)析構(gòu)函數(shù),這樣,C#編譯器才會(huì)生成一個(gè)Finalize方法??梢允褂霉ぞ卟榭碔L代碼得到驗(yàn)證,IL代碼顯示了Finalize方法的結(jié)果是一個(gè)try finally結(jié)果,和using類似。

既然 IDisposable和Finalize都是回收非托管資源,那么為什么微軟要提供雷同的功能呢?


IDisposable和Finalize確實(shí)雷同,但也有一些差別。

Dispose方法是主動(dòng)調(diào)用(使用using也算主動(dòng)調(diào)用),主動(dòng)釋放非托管資源。而析構(gòu)函數(shù)是由GC回收托管資源的時(shí)候調(diào)用,因此何時(shí)調(diào)用是不可掌控的。因此完全依靠析構(gòu)函數(shù)會(huì)導(dǎo)致出現(xiàn)非托管資源遲遲得不到釋放,如果資源過(guò)大,堆積在內(nèi)存增加內(nèi)存泄露的風(fēng)險(xiǎn)。


也許有人會(huì)說(shuō),那我只要把非托管資源放到Dispose方法中,保證釋放,那就不需要使用析構(gòu)函數(shù)了。完全正確。但是前提是你保證釋放了。假設(shè)你寫了一個(gè)框架或者類庫(kù),其中使用了非托管資源,而且也必須顯式調(diào)用Dispose才能釋放掉。但是,假如某個(gè)2B程序員使用這個(gè)類庫(kù),他根本沒(méi)考慮要手動(dòng)釋放資源,因此他在用你的類庫(kù)后,發(fā)現(xiàn)經(jīng)常內(nèi)存動(dòng)不動(dòng)就滿了或者連接數(shù)滿了,然后就開罵你是各種2B,寫的東西是一坨垃圾。


為了避免背這個(gè)黑鍋,也避免被2B罵成2B,你可以依靠析構(gòu)函數(shù)設(shè)置一個(gè)底線,就是在GC啟動(dòng)的時(shí)候也能觸發(fā)一個(gè)動(dòng)作去回收非托管資源,你就需要在Finalize中去做這些事情。


因此對(duì)于釋放非托管資源,可以歸結(jié)2個(gè)要點(diǎn):

1,只實(shí)現(xiàn)Dispose方法,無(wú)法保證人人都會(huì)調(diào)用此方法,萬(wàn)一沒(méi)調(diào)用,背黑鍋的就是你。

2,只實(shí)現(xiàn)析構(gòu)函數(shù),可以保證回收非托管資源,但是無(wú)法掌握釋放時(shí)機(jī),因?yàn)镚C何時(shí)觸發(fā)析構(gòu)函數(shù)是不確定的。運(yùn)氣不好的話,非托管資源長(zhǎng)時(shí)間都得不到釋放。

只有這2個(gè)方法同時(shí)使用,才能最大限度的釋放非托管資源。因此微軟提出了一個(gè) IDisposable 編程模式。具體模式參考http://msdn.microsoft.com/en-us/library/system.idisposable.aspx



在這模式中:

如果用戶調(diào)用了Dispose,則會(huì)調(diào)用Dispose(true),以此表示釋放托管資源和非托管資源。并且注意GC.SuppressFinalize();方法很有必要,這個(gè)方法是壓制了析構(gòu)函數(shù)的調(diào)用,因?yàn)榉峭泄苜Y源以及釋放,析構(gòu)函數(shù)中的釋放語(yǔ)句也沒(méi)有必要執(zhí)行了,一不小心可能還會(huì)導(dǎo)致錯(cuò)誤。


如果用戶沒(méi)有調(diào)用Dispose,那么GC在回收資源的時(shí)候,會(huì)觸發(fā)析構(gòu)函數(shù),注意,此時(shí)傳入的Dispose(false);因?yàn)槲鰳?gòu)函數(shù)只負(fù)責(zé)釋放非托管資源,不建議用析構(gòu)函數(shù)去釋放托管資源,因?yàn)槲鰳?gòu)函數(shù)調(diào)用時(shí)機(jī)是不確定的,托管資源很可能已經(jīng)被釋放掉了,從而導(dǎo)致一些異常的發(fā)生。

采用微軟的模式簡(jiǎn)單有效,當(dāng)然你也可以實(shí)現(xiàn)自己的模式,這就看各人需要了。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI