C# 弱引用WeakReference方法 C#如何实现可被回收的对象引用

WeakReference本质是不阻止GC回收目标对象的引用,与普通强引用不同,它不延长对象生命周期;泛型WeakReference更安全高效,推荐使用。

WeakReference 本质是啥,和普通引用有啥区别

WeakReference 不阻止 GC 回收目标对象,只要没强引用指向它,下一次 GC 就可能被清理掉。普通引用(比如 var obj = new MyClass())会把对象钉在内存里,GC 看到强引用就跳过回收。

关键点在于:WeakReference 本身不延长对象生命周期,但能让你“尝试”访问对象——哪怕它已经被回收了,也不会抛异常,只是 Target 变成 null

怎么安全地用 WeakReference 获取对象并避免空引用异常

不能直接用 Target 做操作,因为读取和使用之间可能已发生 GC。必须用 TryGetTarget(out T result) 原子性判断+获取。

  • TryGetTarget 是线程安全的,且保证返回的 result 非 null 时对象一定还活着
  • 别写 if (wr.Target != null) { wr.Target.DoSomething(); } —— 这中间可能已被回收,Targetnull 后调用会 NRE
  • 如果需要多次访问,应把 TryGetTarget 的结果存到局部变量里再用,而不是反复查 Target
WeakReference wr = new WeakReference(new MyClass());
if (wr.TryGetTarget(out MyClass obj))
{
    obj.DoWork(); // 安全:obj 在这行执行时必然非 null
}
// 此处 obj 是局部变量,不会被 GC

干扰

WeakReference 和 WeakReference 选哪个

WeakReference 是泛型版本,.NET 4.5+ 推荐用它。它省去类型转换、避免装箱(对值类型尤其重要),而且 TryGetTarget 直接返回 T,不用 cast。

  • WeakReference(非泛型):得手动 as T(T)wr.Target,值类型会触发装箱,且可能为 null 即使 T 是 struct(因为 Target 是 object)
  • WeakReferenceTryGetTarget 返回 string,null 安全清晰;WeakReference 也不会装箱
  • 注意:泛型版不支持弱引用数组或 ref 字段,只适用于普通对象引用场景

WeakReference 常见误用场景和坑

它不是缓存方案替代品,也不是“延迟释放”的工具。用错地方反而引发诡异问题。

  • 别把它当“软引用”用:WeakReference 没有保留策略,GC 一来就清,不像 Java 的 SoftReference 会尽量留着
  • 别在静态字典里长期存 WeakReference 却不清理:已回收的条目会堆积,TryGetTarget 返回 false 后应主动从集合中移除
  • 事件订阅用弱引用?不行。WeakReference 无法解决委托持有对象的问题;要用 WeakEventManager 或手动解绑
  • 调试时看到 Target == null 别急着怀疑代码逻辑——可能是 GC 已运行,这是正常行为

真正适合它的场景很窄:比如 UI 控件缓存中防内存泄漏、对象图遍历时避免循环强引用、或实现某些观察者模式中“不阻止被观察者销毁”的关系。用之前先问自己:这个引用是否真的应该不阻碍 GC?