我們平常用的都是對(duì)象的強(qiáng)引用,如果有強(qiáng)引用存在,GC是不會(huì)回收對(duì)象的。我們能不能同時(shí)保持對(duì)對(duì)象的引用,而又可以讓GC需要的時(shí)候回收這個(gè)對(duì)象呢?.NET中提供了WeakReference來(lái)實(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í),可以考慮使用弱引用。弱引用使用起來(lái)很簡(jiǎn)單,看下面的代碼:
復(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)引用。不過(guò)在使用對(duì)象之前要對(duì)其可用性進(jìn)行檢查,因?yàn)樗赡芤呀?jīng)被回收了。如 果你得到的是null(VB.NET下為Nothing),表明對(duì)象已經(jīng)被回收,不能再用了,需要重新分配一個(gè)。如果不是null,就可以放心大膽的用 了。
接下來(lái)讓我們看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ù)活。前面的文章中我提到過(guò)需要Finalize的對(duì)象在最終釋放前會(huì)有一 次復(fù)活,我們大概可以猜到第二個(gè)參數(shù)表示的意思了。如果我們第二個(gè)參數(shù)給false,這個(gè)弱引用就是一個(gè)short weak reference(短弱引用),當(dāng)GC回收時(shí),發(fā)現(xiàn)根中沒(méi)有這個(gè)對(duì)象的引用了,就認(rèn)為這個(gè)對(duì)象無(wú)用,這時(shí)短弱引用對(duì)這個(gè)對(duì)象的跟蹤到此為止,弱引用的 Target被設(shè)置為null。前面的一個(gè)參數(shù)的構(gòu)造函數(shù)版本新建的弱引用為短弱引用。如果第二個(gè)參數(shù)給true,這個(gè)弱引用就是一個(gè)long weak reference(長(zhǎng)弱引用)。在對(duì)象的Finalize方法沒(méi)有被執(zhí)行以前,Target都可用。不過(guò)這是對(duì)象的某些成員變量也許已經(jīng)被回收,所以使 用起來(lái)要想當(dāng)小心。
現(xiàn)在讓我們看看WeakReference是如何實(shí)現(xiàn)的。很顯然WeakReference不能直接的引用目標(biāo)對(duì)象,WeakReference的 Target屬性的get/set是兩個(gè)函數(shù),從某處查到目標(biāo)對(duì)象的引用返回,而不是我們最常用寫(xiě)的那樣直接返回或者設(shè)置一個(gè)私有變量。GC維護(hù)了兩個(gè)列 表來(lái)跟蹤兩種弱引用的目標(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));
}
}
}