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(); }—— 这中间可能已被回收,Target变null后调用会 NRE - 如果需要多次访问,应把
TryGetTarget的结果存到局部变量里再用,而不是反复查Target
WeakReferencewr = 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) - 用
WeakReference:TryGetTarget返回string,null 安全清晰;WeakReference也不会装箱 - 注意:泛型版不支持弱引用数组或 ref 字段,只适用于普通对象引用场景
WeakReference 常见误用场景和坑
它不是缓存方案替代品,也不是“延迟释放”的工具。用错地方反而引发诡异问题。
- 别把它当“软引用”用:
WeakReference没有保留策略,GC 一来就清,不像 Java 的SoftReference会尽量留着 - 别在静态字典里长期存
WeakReference却不清理:已回收的条目会堆积,TryGetTarget返回 false 后应主动从集合中移除 - 事件订阅用弱引用?不行。
WeakReference无法解决委托持有对象的问题;要用WeakEventManager或手动解绑 - 调试时看到
Target == null别急着怀疑代码逻辑——可能是 GC 已运行,这是正常行为
真正适合它的场景很窄:比如 UI 控件缓存中防内存泄漏、对象图遍历时避免循环强引用、或实现某些观察者模式中“不阻止被观察者销毁”的关系。用之前先问自己:这个引用是否真的应该不阻碍 GC?










