Object obj = new Object();
WeakReference wref = new WeakReference( obj );
obj = null;
第一行代碼新建了一個新的對象,這里叫它對象A,obj是對對象A的強引用。接著第二行代碼新建了一個弱引用對象,參數就是對象A的強引用,第三行代碼釋放掉對對象A的強引用。這時如果GC進行回收,對象A就會被回收。
怎樣在取得對象A的強引用呢?很簡單,請看代碼2:
代碼如下:
Object obj2 = wref.Target;
if( obj2 != null )
{
// 做你想做的事吧。
}
else
{
// 對象已經被回收,如果要用必須新建一個。
}
只要顯示的將弱引用的Target屬性附值就會得到弱引用所代表對象的一個強引用。不過在使用對象之前要對其可用性進行檢查,因為它可能已經被回收了。如 果你得到的是null(VB.NET下為Nothing),表明對象已經被回收,不能再用了,需要重新分配一個。如果不是null,就可以放心大膽的用 了。
接下來讓我們看WeakReference的另外一個版本,請看代碼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的另外一個版本有兩個參數,第一個參數和我們前面用的版本的一樣。第二個參數讓我們看一下他的原型,bool trackResurrection,跟蹤復活,是個bool型,就是是否跟蹤復活。前面的文章中我提到過需要Finalize的對象在最終釋放前會有一 次復活,我們大概可以猜到第二個參數表示的意思了。如果我們第二個參數給false,這個弱引用就是一個short weak reference(短弱引用),當GC回收時,發現根中沒有這個對象的引用了,就認為這個對象無用,這時短弱引用對這個對象的跟蹤到此為止,弱引用的 Target被設置為null。前面的一個參數的構造函數版本新建的弱引用為短弱引用。如果第二個參數給true,這個弱引用就是一個long weak reference(長弱引用)。在對象的Finalize方法沒有被執行以前,Target都可用。不過這是對象的某些成員變量也許已經被回收,所以使 用起來要想當小心。現在讓我們看看WeakReference是如何實現的。很顯然WeakReference不能直接的引用目標對象,WeakReference的 Target屬性的get/set是兩個函數,從某處查到目標對象的引用返回,而不是我們最常用寫的那樣直接返回或者設置一個私有變量。GC維護了兩個列 表來跟蹤兩種弱引用的目標對象,在一個 WeakReference對象創建時,它在相應的列表中找到一個位置,將目標對象的引用放入,很顯然,這兩個列表不是根的一部分。在GC進行內存回收的 時候,如果要回收某一個對象,會檢查弱引用的列表,如果保存著這個對象的引用,則將其設為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));
}
}
}
聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com