我們平常用的都是對象的強(qiáng)引用,如果有強(qiáng)引用存在,GC是不會回收對象的。我們能不能同時保持對對象的引用,而又可以讓GC需要的時候回收這個對象呢?.NET中提供了WeakReference來實(shí)現(xiàn)。弱引用可以讓您保持對對象的引用,同時允許GC在必要時釋放對象,回收內(nèi)存。對于那些創(chuàng)建便宜但耗費(fèi)大量內(nèi)存的對象,即希望保持該對象,又要在應(yīng)用程序需要時使用,同時希望GC必要時回收時,可以考慮使用弱引用。弱引用使用起來很簡單,看下面的代碼:
復(fù)制代碼 代碼如下:
Object obj = new Object();
WeakReference wref = new WeakReference( obj );
obj = null;
第一行代碼新建了一個新的對象,這里叫它對象A,obj是對對象A的強(qiáng)引用。接著第二行代碼新建了一個弱引用對象,參數(shù)就是對象A的強(qiáng)引用,第三行代碼釋放掉對對象A的強(qiáng)引用。這時如果GC進(jìn)行回收,對象A就會被回收。
怎樣在取得對象A的強(qiáng)引用呢?很簡單,請看代碼2:
復(fù)制代碼 代碼如下:
Object obj2 = wref.Target;
if( obj2 != null )
{
// 做你想做的事吧。
}
else
{
// 對象已經(jīng)被回收,如果要用必須新建一個。
}
只要顯示的將弱引用的Target屬性附值就會得到弱引用所代表對象的一個強(qiáng)引用。不過在使用對象之前要對其可用性進(jìn)行檢查,因?yàn)樗赡芤呀?jīng)被回收了。如 果你得到的是null(VB.NET下為Nothing),表明對象已經(jīng)被回收,不能再用了,需要重新分配一個。如果不是null,就可以放心大膽的用 了。
接下來讓我們看WeakReference的另外一個版本,請看代碼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的另外一個版本有兩個參數(shù),第一個參數(shù)和我們前面用的版本的一樣。第二個參數(shù)讓我們看一下他的原型,bool trackResurrection,跟蹤復(fù)活,是個bool型,就是是否跟蹤復(fù)活。前面的文章中我提到過需要Finalize的對象在最終釋放前會有一 次復(fù)活,我們大概可以猜到第二個參數(shù)表示的意思了。如果我們第二個參數(shù)給false,這個弱引用就是一個short weak reference(短弱引用),當(dāng)GC回收時,發(fā)現(xiàn)根中沒有這個對象的引用了,就認(rèn)為這個對象無用,這時短弱引用對這個對象的跟蹤到此為止,弱引用的 Target被設(shè)置為null。前面的一個參數(shù)的構(gòu)造函數(shù)版本新建的弱引用為短弱引用。如果第二個參數(shù)給true,這個弱引用就是一個long weak reference(長弱引用)。在對象的Finalize方法沒有被執(zhí)行以前,Target都可用。不過這是對象的某些成員變量也許已經(jīng)被回收,所以使 用起來要想當(dāng)小心。
現(xiàn)在讓我們看看WeakReference是如何實(shí)現(xiàn)的。很顯然WeakReference不能直接的引用目標(biāo)對象,WeakReference的 Target屬性的get/set是兩個函數(shù),從某處查到目標(biāo)對象的引用返回,而不是我們最常用寫的那樣直接返回或者設(shè)置一個私有變量。GC維護(hù)了兩個列 表來跟蹤兩種弱引用的目標(biāo)對象,在一個 WeakReference對象創(chuàng)建時,它在相應(yīng)的列表中找到一個位置,將目標(biāo)對象的引用放入,很顯然,這兩個列表不是根的一部分。在GC進(jìn)行內(nèi)存回收的 時候,如果要回收某一個對象,會檢查弱引用的列表,如果保存著這個對象的引用,則將其設(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));
}
}
}