.net的强引用和弱引用

转自http://www.cnblogs.com/Jessy/p/3594114.html 


一:什么是弱引用

      了解弱引用之前,先了解一下什么是强引用

  例如 : Object obj=new Object();     就是一个强引用,内存分配一份空间给用以存储Object数据,这块内存有一个首地址,也就是obj所保存的数据,内存分配的空间中不仅仅保存着Object对象信息,还保存着自己(Object本身)被引用的次数。

  当一个对象被强引用的形式创建的时候,本身被引用的次数已经为1.

  接着Object o=obj; 这句代码执行之后,obj指向的Object的存储空间已经被引用了2次,所以Object保存的被引用数值为2.

  总结:强引用最终导致的结果就是被引用的对象的被引用次数+1;

  相反的弱引用就是不会对被引用对象的被引用次数有任何影响。

我们平常用的都是对象的强引用,如果有强引用存在,GC是不会回收对象的。我们能不能同时保持对对象的引用,而又可以让GC需要的时候回收这个对象呢?.NET中提供了WeakReference来实现。弱引用可以让您保持对对象的引用,同时允许GC在必要时释放对象,回收内存。对于那些创建便宜但耗费大量内存的对象,即希望保持该对象,又要在应用程序需要时使用,同时希望GC必要时回收时,可以考虑使用弱引用。弱引用使用起来很简单,看下面的代码:

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
{
// 对象已经被回收,如果要用必须新建一个。
}
复制代码

二:弱引用有什么作用

  防止内存泄露。

  Object obj=new Object();

  当你在通过异步的形式访问网络上面的资源的时候,需要的时间可能会比较长,在数据返回之前,用户很可能转向了其他的页面,如果异步访问的对象(obj)对本地的一个对象(Object)是强引用的话,那么在这个异步访问对象(obj)被释放之前,也即在数据被回调之前,这个被引用的对象(Object)是不会被销毁的,这样一来,就导致内存一直被占用。

  WeakReference weakObj=new WeakReference(Object);

  此时就可以使用弱引用,弱引用对象(weakObj)发出异步请求,在回调之前,如果用户要转到其他的页面,这个被引用的对象(Object)是可以被释放的,这样子就不会出现内存一直被占用的现象。

只要显示的将弱引用的Target属性附值就会得到弱引用所代表对象的一个强引用。不过在使用对象之前要对其可用性进行检查,因为它可能已经被回收了。如 果你得到的是null(VB.NET下为Nothing),表明对象已经被回收,不能再用了,需要重新分配一个。如果不是null,就可以放心大胆的用 了。
接下来让我们看WeakReference的另外一个版本,请看代码3:

复制代码
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回收时,发现根中没有这个对象的引用了,就认为这个对象无用,这时短弱引用对这个对象的跟踪到此为止。finalize中如果对象“复活”,short weak reference会忽略,即wr.Target会置null,而long weak reference则会跟踪,wr.Target仍然refer to 原对象,这个是两者的最大区别。如果没有finalize,则两种weak reference表现完全相同。不过这是对象的某些成员变量也许已经被回收,所以使 用起来要想当小心。

现在让我们看看WeakReference是如何实现的。很显然WeakReference不能直接的引用目标对象,WeakReference的 Target属性的get/set是两个函数,从某处查到目标对象的引用返回,而不是我们最常用写的那样直接返回或者设置一个私有变量。GC维护了两个列 表来跟踪两种弱引用的目标对象,在一个 WeakReference对象创建时,它在相应的列表中找到一个位置,将目标对象的引用放入,很显然,这两个列表不是根的一部分。在GC进行内存回收的 时候,如果要回收某一个对象,会检查弱引用的列表,如果保存着这个对象的引用,则将其设为null。 

三:怎样使用弱引用

  弱引用类: WeakReference //有两个重载的构造函数

  WeakReference WeakObj=new WeakReference(Object);//弱引用方式

  IsAlive属性是判断此弱引用对象所引用的对象是否还存在,存在:IsAlive=True;

     Target属性是设置该弱引用对象所引用的数据对象的值


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
强引用弱引用是Java中的两种引用类型它们在内存管理和垃圾回收方面有所不同。 强引用是最常见的引用类型,它会阻止被引用对象被垃圾回收。只要存在强引用指向一个对象,即使系统内存不足,垃圾回收器也不会回收该对象。我们通常使用的对象引用都是强引用。在ThreadLocal中,ThreadLocal对象本身就是一个强引用弱引用是一种比较弱的引用类型,它允许被引用的对象在没有强引用时被垃圾回收。当一个对象只被弱引用所引用时,垃圾回收器在下一次回收时就会将这个对象进行回收,不会等到内存不足的时候。在ThreadLocal中,ThreadLocalMap的Entry对象的key就是弱引用,即ThreadLocal对象的弱引用。 ThreadLocal使用弱引用作为key的原因是为了避免内存泄漏。强引用的话,即使ThreadLocal对象已经没有被使用,但是由于强引用仍然存在,ThreadLocal对象无法被垃圾回收,从而导致内存泄漏。通过使用弱引用,当ThreadLocal对象没有其他强引用时,就可以被垃圾回收,避免内存泄漏的问题。 如果不使用弱引用,我们可以将ThreadLocal对象设置为null,手动释放对ThreadLocal的引用,从而让ThreadLocal对象能够被垃圾回收。但是这种方式需要我们手动管理,很容易出错。使用弱引用可以让垃圾回收器自动回收不再使用的ThreadLocal对象,更加方便和安全。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [强引用弱引用的Threadlocal](https://blog.csdn.net/a779868946/article/details/121458153)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *3* [ThreadLocal之强、弱、软、虚引用](https://blog.csdn.net/weixin_43847283/article/details/125470183)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值