在 .NET 中,弱引用(WeakReference)是一种特殊的引用类型,它允许你引用一个对象而不阻止该对象被垃圾回收器(GC)回收。与强引用不同,强引用会延长对象的生命周期,只要还有强引用存在,对象就不会被回收;而弱引用不会影响对象的生命周期。
这种机制特别适用于缓存场景——你希望保留对某些对象的引用以提升性能,但又不希望这些引用导致内存无法释放,从而引发内存泄漏。
为什么缓存容易导致内存泄漏?
假设你使用一个静态字典来缓存大量对象:
private static Dictionary<string, HeavyObject> _cache = new();
每次请求都检查这个字典,如果存在就返回缓存对象。问题在于:只要对象在字典中,它就有强引用,即使程序其他地方已不再需要它,GC 也无法回收。随着缓存增长,内存持续上升,最终可能导致 OutOfMemoryException。
如何用 WeakReference 解决缓存内存泄漏?
你可以将缓存中的值从直接存储对象改为存储 WeakReference,这样对象可以被 GC 回收,同时你还能尝试访问它(如果还活着)。
示例:使用 WeakReference 实现弱缓存
private static Dictionary<string, WeakReference<HeavyObject>> _weakCache = new();
public static HeavyObject GetFromCache(string key)
{
if (_weakCache.TryGetValue(key, out var weakRef))
{
// 尝试获取目标对象
if (weakRef.TryGetTarget(out var target))
{
return target; // 对象仍存活,直接返回
}
else
{
// 对象已被回收,从缓存中移除
_weakCache.Remove(key);
}
}
// 缓存未命中或对象已回收,重新创建
var newObj = new HeavyObject();
_weakCache[key] = new WeakReference<HeavyObject>(newObj);
return newObj;
}
在这个例子中:
缓存保存的是WeakReference<heavyobject></heavyobject>,不是对象本身。 通过
TryGetTarget()检查对象是否还活着。 如果对象已被回收,清理缓存条目并重建。
WeakReference 的两种模式
.NET 提供了两种弱引用模式:
短弱引用(Short Weak Reference):只跟踪对象是否存活。一旦对象被回收,引用即失效。上面的例子就是短弱引用。 长弱引用(Long Weak Reference):即使对象已经执行了终结器(finalizer),仍可追踪其引用(前提是对象没有被压缩或移动)。使用时需设置trackResurrection = true。
大多数缓存场景使用短弱引用就够了。
实际建议和注意事项
弱引用适合可再生资源的缓存,比如计算结果、临时数据等,不能用于必须长期持有的对象。 频繁检查TryGetTarget()返回 false 时应清理缓存,避免“僵尸”条目堆积。 考虑结合 ConditionalWeakTable 或第三方库如 MemoryCache 来实现更高级的缓存策略。 注意性能:弱引用本身有轻微开销,不适合超高频访问的场景。
基本上就这些。用好 WeakReference,可以在提升性能的同时避免内存失控。关键是要理解:它不保证对象还在,只是“试着看看还在不在”。
