๐ Introduction
In Python, memory management is mostly handled by the garbage collector. But in some advanced use cases, we need to keep track of objects without preventing them from being garbage-collected.
This is where weak references come into play.
In this post, weโll explore:
โ What weak references are and how they work
โ How they help avoid memory leaks
โ When and why you should use them
โ Practical examples using weakref
โ Best practices for using weak references
1๏ธโฃ What Are Weak References?
A weak reference is a reference to an object that doesnโt increase its reference count. This means:
If there are only weak references to an object, it can still be garbage-collected.
Weak references are useful for caching, tracking, or observing objects without keeping them alive.
๐น Contrast with a normal (strong) reference:
a = SomeObject() # Strong reference โ keeps object alive
๐น With weak reference:
import weakref
obj = SomeObject()
w = weakref.ref(obj) # Weak reference
print(w()) # Returns the object while it's alive
del obj
print(w()) # Returns None if object is collected
2๏ธโฃ Why Use Weak References?
โ To avoid memory leaks in long-running applications
โ To build caches that auto-expire when the objects are no longer used
โ To track objects without preventing their destruction
โ To store metadata about objects without owning them
3๏ธโฃ Using weakref.ref() โ The Basics
import weakref
class Person:
def __init__(self, name):
self.name = name
p = Person("Alice")
weak_p = weakref.ref(p)
print(weak_p) # <weakref at 0x...; to 'Person' at 0x...>
print(weak_p()) # <__main__.Person object at ...>
print(weak_p().name) # Alice
del p
print(weak_p()) # None โ object has been garbage collected
โ Key point: The weak reference does not prevent garbage collection.
4๏ธโฃ Using weakref.WeakKeyDictionary
A dictionary where keys are stored as weak references. When the object is deleted, the key is removed automatically.
import weakref
class Data:
def __init__(self, value):
self.value = value
data1 = Data(10)
data2 = Data(20)
wk_dict = weakref.WeakKeyDictionary()
wk_dict[data1] = "Object 1"
wk_dict[data2] = "Object 2"
print(dict(wk_dict)) # {data1: ..., data2: ...}
del data1
print(dict(wk_dict)) # {data2: ...} โ data1 key removed automatically
โ Use Case: Managing metadata without holding onto large objects.
5๏ธโฃ Using weakref.WeakValueDictionary
A dictionary where values are weak references.
import weakref
class User:
def __init__(self, username):
self.username = username
u1 = User("alice")
u2 = User("bob")
users = weakref.WeakValueDictionary()
users["a"] = u1
users["b"] = u2
print(users["a"].username) # alice
del u1
print("a" in users) # False โ value was garbage-collected
โ Use Case: Auto-expiring caches for objects that shouldnโt stay in memory forever.
6๏ธโฃ Weak References with Callbacks
You can attach a callback to a weak reference. This is useful for cleanup or logging.
import weakref
class Item:
def __del__(self):
print("Item deleted")
def on_finalize(wr):
print("Object has been garbage collected!")
item = Item()
ref = weakref.ref(item, on_finalize)
del item
# Output:
# Item deleted
# Object has been garbage collected!
โ Use Case: Perform cleanup when objects are collected.
7๏ธโฃ Common Pitfalls and Tips
โ Accessing a dead weak reference
obj = SomeObject()
w = weakref.ref(obj)
del obj
print(w()) # None โ object no longer exists
โ๏ธ Always check if the weak reference is still valid before using it.
โ Trying to weak-reference an unsupported object
Not all objects can be weakly referenced. For example, int, list, str, and dict usually donโt support weak references.
import weakref
w = weakref.ref(42) # โ TypeError: cannot create weak reference to 'int' object
โ๏ธ Only class instances (and custom objects) typically support weak references.
โ Best Practices
Use weak references when you donโt want to own the object.
Avoid memory leaks in observer patterns, plugin systems, and caching layers.
Always check ref() is not None before accessing the object.
8๏ธโฃ Summary
โ๏ธ Weak references donโt increase reference counts.
โ๏ธ Used to track objects without preventing their collection.
โ๏ธ WeakKeyDictionary and WeakValueDictionary manage memory-friendly mappings.
โ๏ธ Callbacks can be attached to perform cleanup when objects are deleted.
โ๏ธ Ideal for caches, plugins, and memory-sensitive apps.
๐ Whatโs Next?
Next, we'll explore "Mastering Comprehensions in Python โ The Pythonic Way to Build Data Structures" Stay tuned.
Top comments (0)