通常的,當(dāng)說起對象引用的時候,一般指的是強(qiáng)引用,即只要這個對象還是可達(dá)狀態(tài)(還會被程序訪問到),那么垃圾回收器就不會去回收它。
廣信網(wǎng)站制作公司哪家好,找創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、自適應(yīng)網(wǎng)站建設(shè)等網(wǎng)站項目制作,到程序開發(fā),運(yùn)營維護(hù)。創(chuàng)新互聯(lián)于2013年成立到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運(yùn)維經(jīng)驗,來保證我們的工作的順利進(jìn)行。專注于網(wǎng)站建設(shè)就選創(chuàng)新互聯(lián)。
而弱引用的對象被認(rèn)為是不可達(dá)的,但它可以由應(yīng)用程序訪問,同時還能被垃圾回收器收回。支持垃圾收集的語言大多都支持弱引用,例如Java,C#,Python等。
通常,弱引用特別適合以下對象:占用大量內(nèi)存,但通過垃圾回收功能回收以后很容易重新創(chuàng)建。C#中使用WeakReference類來創(chuàng)建弱引用對象。
下面是示例代碼:
public string FilePath = "PathToMyImportantFile.dat"; public WeakReference WeakRef = new WeakReference(null); public object ImportantBigFileContents { get { object bigObject = WeakRef.Target; if (bigObject == null)// 該弱引用已經(jīng)被回收了,那么就變成null,則需要構(gòu)造大對象。 { using (StreamReader r = new StreamReader(FilePath)) bigObject = r.ReadToEnd(); WeakRef.Target = bigObject; } return bigObject; } }
假設(shè)ImportantBigFileContents是某一個類的屬性,該屬性值是一個很占內(nèi)存的對象,那么可以考慮使用弱引用存儲這個大對象。
如果弱引用中的Target屬性中的值仍然存在,則直接獲取這個值返回。如果這個值已經(jīng)被垃圾回收器回收掉了,那么這個值就是null。因此需要重新構(gòu)造出該大對象。
注意,垃圾回收器究竟再何時啟動,程序員是沒法掌控的,因此也不能確定弱引用對象何時被回收。所以,很容易犯如下的錯誤,比如如下代碼:
public object ImportantBigFileContents { get { if (WeakRef.Target == null) using (StreamReader r = new StreamReader(FilePath)) WeakRef.Target = r.ReadToEnd(); return WeakRef.Target; } }
看上去沒啥錯誤,如果WeakRef.Target屬性為null則構(gòu)造出大對象,如果WeakRef.Target有值則返回該值。
但是這里有一個小概率事情,由于垃圾回收器是異步執(zhí)行的,你不會知道啥時回收,有一種可能是要調(diào)用return WeakRef.Target;句子前回收了。這樣的話就會導(dǎo)致返回null。因此,推薦的做法是在讀取弱引用對象之前,還是把它放入強(qiáng)引用對象中,即放入一個普通的對象中去。
弱引用還可以用來處理內(nèi)存泄露的風(fēng)險,例如常見的垃圾回收算法是引用計數(shù),引用計數(shù)法計算了對象被引用的次數(shù),在被引用的次數(shù)為0的時候,回收該對象。但是對于環(huán)形引用的對象,無法回收。即比如A中對象引用了B對象,B對象中也引用了A對象,這種情況下,垃圾回收器無法回收A對象和B對象,一旦這種環(huán)形引用的多了之后,就會引發(fā)內(nèi)存泄露。
如果把A對象和B對象都設(shè)成弱引用的話,GC在必要的時候還是會收回資源的。
但是同時也要避免對小對象使用弱引用,因為指針本身可能和對象一樣大,或者比對象還大。事實上,如果內(nèi)存不是那么緊張的話,也沒必要過度的使用弱引用。