WeakReference(弱引用)讓GC需要時(shí)回收對(duì)象
更新時(shí)間:2012年12月21日 11:54:41 作者:
我們平常用的都是對(duì)象的強(qiáng)引用,如果有強(qiáng)引用存在,GC是不會(huì)回收對(duì)象的,我們能不能同時(shí)保持對(duì)對(duì)象的引用,而又可以讓GC需要的時(shí)候回收這個(gè)對(duì)象呢?本文將為您詳細(xì)解答,需要了解的朋友可以參考下
我們平常用的都是對(duì)象的強(qiáng)引用,如果有強(qiáng)引用存在,GC是不會(huì)回收對(duì)象的。我們能不能同時(shí)保持對(duì)對(duì)象的引用,而又可以讓GC需要的時(shí)候回收這個(gè)對(duì)象呢?.NET中提供了WeakReference來實(shí)現(xiàn)。弱引用可以讓您保持對(duì)對(duì)象的引用,同時(shí)允許GC在必要時(shí)釋放對(duì)象,回收內(nèi)存。對(duì)于那些創(chuàng)建便宜但耗費(fèi)大量?jī)?nèi)存的對(duì)象,即希望保持該對(duì)象,又要在應(yīng)用程序需要時(shí)使用,同時(shí)希望GC必要時(shí)回收時(shí),可以考慮使用弱引用。弱引用使用起來很簡(jiǎn)單,看下面的代碼:
Object obj = new Object();
WeakReference wref = new WeakReference( obj );
obj = null;
第一行代碼新建了一個(gè)新的對(duì)象,這里叫它對(duì)象A,obj是對(duì)對(duì)象A的強(qiáng)引用。接著第二行代碼新建了一個(gè)弱引用對(duì)象,參數(shù)就是對(duì)象A的強(qiáng)引用,第三行代碼釋放掉對(duì)對(duì)象A的強(qiáng)引用。這時(shí)如果GC進(jìn)行回收,對(duì)象A就會(huì)被回收。
怎樣在取得對(duì)象A的強(qiáng)引用呢?很簡(jiǎn)單,請(qǐng)看代碼2:
Object obj2 = wref.Target;
if( obj2 != null )
{
// 做你想做的事吧。
}
else
{
// 對(duì)象已經(jīng)被回收,如果要用必須新建一個(gè)。
}
只要顯示的將弱引用的Target屬性附值就會(huì)得到弱引用所代表對(duì)象的一個(gè)強(qiáng)引用。不過在使用對(duì)象之前要對(duì)其可用性進(jìn)行檢查,因?yàn)樗赡芤呀?jīng)被回收了。如 果你得到的是null(VB.NET下為Nothing),表明對(duì)象已經(jīng)被回收,不能再用了,需要重新分配一個(gè)。如果不是null,就可以放心大膽的用 了。
接下來讓我們看WeakReference的另外一個(gè)版本,請(qǐng)看代碼3:
// public WeakReference(
// object target,
// bool trackResurrection
//);
Object obj1 = new Object();
Object obj2 = new Object();
WeakReference wref1 = new WeakReference( obj1, false );
WeakReference wref2 = new WeakReference( obj2, true );
WeakReference的另外一個(gè)版本有兩個(gè)參數(shù),第一個(gè)參數(shù)和我們前面用的版本的一樣。第二個(gè)參數(shù)讓我們看一下他的原型,bool trackResurrection,跟蹤復(fù)活,是個(gè)bool型,就是是否跟蹤復(fù)活。前面的文章中我提到過需要Finalize的對(duì)象在最終釋放前會(huì)有一 次復(fù)活,我們大概可以猜到第二個(gè)參數(shù)表示的意思了。如果我們第二個(gè)參數(shù)給false,這個(gè)弱引用就是一個(gè)short weak reference(短弱引用),當(dāng)GC回收時(shí),發(fā)現(xiàn)根中沒有這個(gè)對(duì)象的引用了,就認(rèn)為這個(gè)對(duì)象無用,這時(shí)短弱引用對(duì)這個(gè)對(duì)象的跟蹤到此為止,弱引用的 Target被設(shè)置為null。前面的一個(gè)參數(shù)的構(gòu)造函數(shù)版本新建的弱引用為短弱引用。如果第二個(gè)參數(shù)給true,這個(gè)弱引用就是一個(gè)long weak reference(長(zhǎng)弱引用)。在對(duì)象的Finalize方法沒有被執(zhí)行以前,Target都可用。不過這是對(duì)象的某些成員變量也許已經(jīng)被回收,所以使 用起來要想當(dāng)小心。
現(xiàn)在讓我們看看WeakReference是如何實(shí)現(xiàn)的。很顯然WeakReference不能直接的引用目標(biāo)對(duì)象,WeakReference的 Target屬性的get/set是兩個(gè)函數(shù),從某處查到目標(biāo)對(duì)象的引用返回,而不是我們最常用寫的那樣直接返回或者設(shè)置一個(gè)私有變量。GC維護(hù)了兩個(gè)列 表來跟蹤兩種弱引用的目標(biāo)對(duì)象,在一個(gè) WeakReference對(duì)象創(chuàng)建時(shí),它在相應(yīng)的列表中找到一個(gè)位置,將目標(biāo)對(duì)象的引用放入,很顯然,這兩個(gè)列表不是根的一部分。在GC進(jìn)行內(nèi)存回收的 時(shí)候,如果要回收某一個(gè)對(duì)象,會(huì)檢查弱引用的列表,如果保存著這個(gè)對(duì)象的引用,則將其設(shè)為null。
public class AspPage : Page
{
private static ArrayList __ENCList = new ArrayList();
[DebuggerNonUserCode]
public AspPage()
{
base.Load += new EventHandler(this.Page_Load);
ArrayList list = __ENCList;
lock (list)
{
__ENCList.Add(new WeakReference(this));
}
}
}
復(fù)制代碼 代碼如下:
Object obj = new Object();
WeakReference wref = new WeakReference( obj );
obj = null;
第一行代碼新建了一個(gè)新的對(duì)象,這里叫它對(duì)象A,obj是對(duì)對(duì)象A的強(qiáng)引用。接著第二行代碼新建了一個(gè)弱引用對(duì)象,參數(shù)就是對(duì)象A的強(qiáng)引用,第三行代碼釋放掉對(duì)對(duì)象A的強(qiáng)引用。這時(shí)如果GC進(jìn)行回收,對(duì)象A就會(huì)被回收。
怎樣在取得對(duì)象A的強(qiáng)引用呢?很簡(jiǎn)單,請(qǐng)看代碼2:
復(fù)制代碼 代碼如下:
Object obj2 = wref.Target;
if( obj2 != null )
{
// 做你想做的事吧。
}
else
{
// 對(duì)象已經(jīng)被回收,如果要用必須新建一個(gè)。
}
只要顯示的將弱引用的Target屬性附值就會(huì)得到弱引用所代表對(duì)象的一個(gè)強(qiáng)引用。不過在使用對(duì)象之前要對(duì)其可用性進(jìn)行檢查,因?yàn)樗赡芤呀?jīng)被回收了。如 果你得到的是null(VB.NET下為Nothing),表明對(duì)象已經(jīng)被回收,不能再用了,需要重新分配一個(gè)。如果不是null,就可以放心大膽的用 了。
接下來讓我們看WeakReference的另外一個(gè)版本,請(qǐng)看代碼3:
復(fù)制代碼 代碼如下:
// public WeakReference(
// object target,
// bool trackResurrection
//);
Object obj1 = new Object();
Object obj2 = new Object();
WeakReference wref1 = new WeakReference( obj1, false );
WeakReference wref2 = new WeakReference( obj2, true );
WeakReference的另外一個(gè)版本有兩個(gè)參數(shù),第一個(gè)參數(shù)和我們前面用的版本的一樣。第二個(gè)參數(shù)讓我們看一下他的原型,bool trackResurrection,跟蹤復(fù)活,是個(gè)bool型,就是是否跟蹤復(fù)活。前面的文章中我提到過需要Finalize的對(duì)象在最終釋放前會(huì)有一 次復(fù)活,我們大概可以猜到第二個(gè)參數(shù)表示的意思了。如果我們第二個(gè)參數(shù)給false,這個(gè)弱引用就是一個(gè)short weak reference(短弱引用),當(dāng)GC回收時(shí),發(fā)現(xiàn)根中沒有這個(gè)對(duì)象的引用了,就認(rèn)為這個(gè)對(duì)象無用,這時(shí)短弱引用對(duì)這個(gè)對(duì)象的跟蹤到此為止,弱引用的 Target被設(shè)置為null。前面的一個(gè)參數(shù)的構(gòu)造函數(shù)版本新建的弱引用為短弱引用。如果第二個(gè)參數(shù)給true,這個(gè)弱引用就是一個(gè)long weak reference(長(zhǎng)弱引用)。在對(duì)象的Finalize方法沒有被執(zhí)行以前,Target都可用。不過這是對(duì)象的某些成員變量也許已經(jīng)被回收,所以使 用起來要想當(dāng)小心。
現(xiàn)在讓我們看看WeakReference是如何實(shí)現(xiàn)的。很顯然WeakReference不能直接的引用目標(biāo)對(duì)象,WeakReference的 Target屬性的get/set是兩個(gè)函數(shù),從某處查到目標(biāo)對(duì)象的引用返回,而不是我們最常用寫的那樣直接返回或者設(shè)置一個(gè)私有變量。GC維護(hù)了兩個(gè)列 表來跟蹤兩種弱引用的目標(biāo)對(duì)象,在一個(gè) WeakReference對(duì)象創(chuàng)建時(shí),它在相應(yīng)的列表中找到一個(gè)位置,將目標(biāo)對(duì)象的引用放入,很顯然,這兩個(gè)列表不是根的一部分。在GC進(jìn)行內(nèi)存回收的 時(shí)候,如果要回收某一個(gè)對(duì)象,會(huì)檢查弱引用的列表,如果保存著這個(gè)對(duì)象的引用,則將其設(shè)為null。
復(fù)制代碼 代碼如下:
public class AspPage : Page
{
private static ArrayList __ENCList = new ArrayList();
[DebuggerNonUserCode]
public AspPage()
{
base.Load += new EventHandler(this.Page_Load);
ArrayList list = __ENCList;
lock (list)
{
__ENCList.Add(new WeakReference(this));
}
}
}
相關(guān)文章
ASP.NET中Web API的簡(jiǎn)單實(shí)例
Web API框架是一個(gè)面向Http協(xié)議的通信框架,Web API 框架是一個(gè)面向Http協(xié)議的通信框架。Web API 框架目前支持兩種數(shù)據(jù)格式的序列化:Json 及 Xml。在不做任何配置的情況下,則 Web API 會(huì)自動(dòng)把數(shù)據(jù)使用xml進(jìn)行序列化,否則使用 json 序列化,需要的朋友可以參考下2015-10-10EFCore 通過實(shí)體Model生成創(chuàng)建SQL Server數(shù)據(jù)庫(kù)表腳本
這篇文章主要介紹了EFCore 通過實(shí)體Model生成創(chuàng)建SQL Server數(shù)據(jù)庫(kù)表腳本的示例,幫助大家更好的理解和學(xué)習(xí)使用.net框架,感興趣的朋友可以了解下2021-03-03ASP.NET Core 2.0 帶初始參數(shù)的中間件問題及解決方法
這篇文章主要介紹了ASP.NET Core 2.0 帶初始參數(shù)的中間件問題及解決方法,需要的朋友可以參考下2017-10-10ASP.NET Gridview 中使用checkbox刪除的2種方法實(shí)例分享
ASP.NET Gridview 中使用checkbox刪除的2種方法實(shí)例分享,需要的朋友可以參考一下2013-06-06.NET Core中創(chuàng)建和使用NuGet包的示例代碼
這篇文章主要介紹了.NET Core中創(chuàng)建和使用NuGet包的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04ASP.NET中控件的EnableViewState屬性及徹底禁用
如果我們?cè)陂_發(fā)Web應(yīng)用程序時(shí),某些控件是不需要接受用戶的操作或只需要接受一次操作的時(shí)候,我們可以將這些控件的EnableViewState屬性改為false,這樣可以優(yōu)化我們的程序,提高網(wǎng)絡(luò)訪問的速度。2016-06-06