该用weakref当且仅当你持有对象但不想阻止其被垃圾回收。它适用于缓存、观察者模式、对象池等场景,通过弱引用避免循环引用和内存泄漏,常用类型包括weakref.ref、WeakKeyDictionary、WeakValueDictionary和finalize。
Python的weakref模块用于创建弱引用,避免循环引用导致的对象无法被垃圾回收,从而节省内存。它不增加对象的引用计数,因此当原对象被销毁时,弱引用自动失效(变为None或触发回调),特别适合缓存、观察者模式、对象池等场景。
核心判断标准:你持有某个对象,但**不想阻止它被回收**。
weakref.WeakValueDictionary)weakref.finalize
,造成内存泄漏最常用的是weakref.ref和基于它的容器类,它们行为明确、开销小。
weakref.ref(obj):返回一个可调用对象,调用后得到原对象(若还存在),否则返回None
weakref.WeakKeyDictionary():键是弱引用,键对象被回收后,对应条目自动删除weakref.WeakValueDictionary():值是弱引用,值对象被回收后,条目自动清除weakref.finalize(obj, callback, *args):在obj被销毁前执行callback,适合资源清理示例:
import weakrefclass Data: def init(self, name): self.name = name def del(self): print(f"{self.name} 被销毁")
obj = Data("cache_item") wref = weakref.ref(obj) print(wref()) # 输出: <main.Data object at 0x...> del obj print(wref()) # 输出: None(对象已销毁)
相比普通字典,它能自动清理已销毁的对象,避免“幽灵缓存”。
int、str),值必须支持弱引用(不能是list、dict、int等内置不可弱引用类型)示例:
cache = weakref.WeakValueDictionary()def getdata(id): if id not in cache: cache[id] = Data(f"item{id}") # 值被弱引用 return cache[id]
a = get_data(100) print(100 in cache) # True del a
此时cache[100] 已自动移除(下一次访问会重建)
弱引用不是万能的,用错反而引入隐性bug。
int、str、list、dict等默认不支持__weakref__,尝试创建弱引用会报TypeError
ref存进全局列表,就又变成强引用了wref() is not None和后续使用之间,对象可能已被回收;建议用try/except ReferenceError捕获__del__或显式close)不复杂但容易忽略。