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
WeakReference<MyClass> wr = new WeakReference<MyClass>(new MyClass());
if (wr.TryGetTarget(out MyClass obj))
{
obj.DoWork(); // 安全:obj 在这行执行时必然非 null
}
// 此处 obj 是局部变量,不会被 GC 干扰
WeakReference 和 WeakReference 选哪个
WeakReference<t></t>是泛型版本,.NET 4.5+ 推荐用它。它省去类型转换、避免装箱(对值类型尤其重要),而且
TryGetTarget直接返回
T,不用 cast。 用
WeakReference(非泛型):得手动
as T或
(T)wr.Target,值类型会触发装箱,且可能为
null即使 T 是 struct(因为 Target 是 object) 用
WeakReference<string></string>:
TryGetTarget返回
string,null 安全清晰;
WeakReference<int></int>也不会装箱 注意:泛型版不支持弱引用数组或 ref 字段,只适用于普通对象引用场景
WeakReference 常见误用场景和坑
它不是缓存方案替代品,也不是“延迟释放”的工具。用错地方反而引发诡异问题。
别把它当“软引用”用:WeakReference没有保留策略,GC 一来就清,不像 Java 的
SoftReference会尽量留着 别在静态字典里长期存
WeakReference却不清理:已回收的条目会堆积,
TryGetTarget返回 false 后应主动从集合中移除 事件订阅用弱引用?不行。
WeakReference无法解决委托持有对象的问题;要用
WeakEventManager或手动解绑 调试时看到
Target == null别急着怀疑代码逻辑——可能是 GC 已运行,这是正常行为
真正适合它的场景很窄:比如 UI 控件缓存中防内存泄漏、对象图遍历时避免循环强引用、或实现某些观察者模式中“不阻止被观察者销毁”的关系。用之前先问自己:这个引用是否真的应该不阻碍 GC?
