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

来源:这里教程网 时间:2026-02-21 17:39:17 作者:

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?

相关推荐