[code.view]

[top] / python / PyMOTW / weakref / index.rst

     ====================================================
     weakref -- Garbage-collectable references to objects
     ====================================================
     
     .. module:: weakref
         :synopsis: Refer to an "expensive" object, but allow it to be garbage collected if there are no other non-weak references.
     
     :Purpose: Refer to an "expensive" object, but allow it to be garbage collected if there are no other non-weak references.
     :Python Version: Since 2.1
     
     The :mod:`weakref` module supports weak references to objects. A
     normal reference increments the reference count on the object and
     prevents it from being garbage collected. This is not always
     desirable, either when a circular reference might be present or when
     building a cache of objects that should be deleted when memory is
     needed.
     
     References
     ==========
     
     Weak references to your objects are managed through the :class:`ref`
     class. To retrieve the original object, call the reference object.
     
     .. include:: weakref_ref.py
         :literal:
         :start-after: #end_pymotw_header
     
     In this case, since ``obj`` is deleted before the second call to the
     reference, the :class:`ref` returns ``None``.
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, 'weakref_ref.py'))
     .. }}}
     
     ::
     
     	$ python weakref_ref.py
     
     	obj: <__main__.ExpensiveObject object at 0x100464790>
     	ref: <weakref at 0x100458ba8; to 'ExpensiveObject' at 0x100464790>
     	r(): <__main__.ExpensiveObject object at 0x100464790>
     	deleting obj
     	(Deleting <__main__.ExpensiveObject object at 0x100464790>)
     	r(): None
     
     .. {{{end}}}
     
     Reference Callbacks
     ===================
     
     The :class:`ref` constructor takes an optional second argument that
     should be a callback function to invoke when the referenced object is
     deleted.
     
     .. include:: weakref_ref_callback.py
         :literal:
         :start-after: #end_pymotw_header
     
     The callback receives the reference object as an argument, after the
     reference is "dead" and no longer refers to the original object. This
     lets you remove the weak reference object from a cache, for example.
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, 'weakref_ref_callback.py'))
     .. }}}
     
     ::
     
     	$ python weakref_ref_callback.py
     
     	obj: <__main__.ExpensiveObject object at 0x100462990>
     	ref: <weakref at 0x100458c00; to 'ExpensiveObject' at 0x100462990>
     	r(): <__main__.ExpensiveObject object at 0x100462990>
     	deleting obj
     	callback( <weakref at 0x100458c00; dead> )
     	(Deleting <__main__.ExpensiveObject object at 0x100462990>)
     	r(): None
     
     .. {{{end}}}
     
     Proxies
     =======
     
     Instead of using :class:`ref` directly, it can be more convenient to
     use a proxy.  Proxies can be used as though they were the original
     object, so you do not need to call the :class:`ref` first to access
     the object.
     
     .. include:: weakref_proxy.py
         :literal:
         :start-after: #end_pymotw_header
     
     If the proxy is access after the referent object is removed, a
     :class:`ReferenceError` exception is raised.
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, 'weakref_proxy.py', ignore_error=True))
     .. }}}
     
     ::
     
     	$ python weakref_proxy.py
     
     	via obj: My Object
     	via ref: My Object
     	via proxy: My Object
     	(Deleting <__main__.ExpensiveObject object at 0x100462810>)
     	via proxy:
     	Traceback (most recent call last):
     	  File "weakref_proxy.py", line 26, in <module>
     	    print 'via proxy:', p.name
     	ReferenceError: weakly-referenced object no longer exists
     
     .. {{{end}}}
     
     Cyclic References
     =================
     
     One use for weak references is to allow cyclic references without preventing
     garbage collection. This example illustrates the difference between using
     regular objects and proxies when a graph includes a cycle.
     
     First, we need a :class:`Graph` class that accepts any object given to
     it as the "next" node in the sequence. For the sake of brevity, this
     :class:`Graph` supports a single outgoing reference from each node,
     which results in boring graphs but makes it easy to create cycles. The
     function :func:`demo()` is a utility function to exercise the graph
     class by creating a cycle and then removing various references.
     
     .. include:: weakref_graph.py
        :literal:
        :start-after: #end_pymotw_header
     
     Now we can set up a test program using the :mod:`gc` module to help us
     debug the leak. The ``DEBUG_LEAK`` flag causes :mod:`gc` to print
     information about objects that cannot be seen other than through the
     reference the garbage collector has to them.
     
     .. include:: weakref_cycle.py
        :literal:
        :start-after: #end_pymotw_header
     
     Even after deleting the local references to the :class:`Graph`
     instances in :func:`demo()`, the graphs all show up in the garbage
     list and cannot be collected.  The dictionaries in the garbage list
     hold the attributes of the :class:`Graph` instances. We can forcibly
     delete the graphs, since we know what they are:
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, '-u weakref_cycle.py'))
     .. }}}
     
     ::
     
     	$ python -u weakref_cycle.py
     
     	Setting up the cycle
     	
     	Set up graph:
     	one.set_next(two (<class 'weakref_graph.Graph'>))
     	two.set_next(three (<class 'weakref_graph.Graph'>))
     	three.set_next(one->two->three (<class 'weakref_graph.Graph'>))
     	
     	Graphs:
     	one->two->three->one
     	two->three->one->two
     	three->one->two->three
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     	
     	After 2 references removed:
     	one->two->three->one
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     	
     	Removing last reference:
     	Collecting...
     	gc: uncollectable <Graph 0x100477590>
     	gc: uncollectable <Graph 0x1004775d0>
     	gc: uncollectable <Graph 0x100477610>
     	gc: uncollectable <dict 0x10035fb00>
     	gc: uncollectable <dict 0x1003612e0>
     	gc: uncollectable <dict 0x10035c210>
     	Unreachable objects: 6
     	Garbage:[Graph(one),
     	 Graph(two),
     	 Graph(three),
     	 {'name': 'one', 'other': Graph(two)},
     	 {'name': 'two', 'other': Graph(three)},
     	 {'name': 'three', 'other': Graph(one)}]
     	
     	Breaking the cycle and cleaning up garbage
     	
     	one.set_next(None (<type 'NoneType'>))
     	(Deleting two)
     	two.set_next(None (<type 'NoneType'>))
     	(Deleting three)
     	three.set_next(None (<type 'NoneType'>))
     	(Deleting one)
     	one.set_next(None (<type 'NoneType'>))
     	
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     
     .. {{{end}}}
     
     
     And now let's define a more intelligent :class:`WeakGraph` class that
     knows not to create cycles using regular references, but to use a
     :class:`ref` when a cycle is detected.
     
     .. include:: weakref_weakgraph.py
        :literal:
        :start-after: #end_pymotw_header
     
     Since the :class:`WeakGraph` instances use proxies to refer to objects
     that have already been seen, as :func:`demo()` removes all local
     references to the objects, the cycle is broken and the garbage
     collector can delete the objects for us.
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, 'weakref_weakgraph.py'))
     .. }}}
     
     ::
     
     	$ python weakref_weakgraph.py
     
     	Set up graph:
     	one.set_next(two (<class '__main__.WeakGraph'>))
     	two.set_next(three (<class '__main__.WeakGraph'>))
     	three.set_next(one->two->three (<type 'weakproxy'>))
     	
     	Graphs:
     	one->two->three
     	two->three->one->two
     	three->one->two->three
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     	
     	After 2 references removed:
     	one->two->three
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     	
     	Removing last reference:
     	(Deleting one)
     	one.set_next(None (<type 'NoneType'>))
     	(Deleting two)
     	two.set_next(None (<type 'NoneType'>))
     	(Deleting three)
     	three.set_next(None (<type 'NoneType'>))
     	Collecting...
     	Unreachable objects: 0
     	Garbage:[]
     
     .. {{{end}}}
     
     
     Caching Objects
     ===============
     
     The :class:`ref` and :class:`proxy` classes are considered "low
     level". While they are useful for maintaining weak references to
     individual objects and allowing cycles to be garbage collected, if you
     need to create a cache of several objects the
     :class:`WeakKeyDictionary` and :class:`WeakValueDictionary` provide a
     more appropriate API.
     
     As you might expect, the :class:`WeakValueDictionary` uses weak
     references to the values it holds, allowing them to be garbage
     collected when other code is not actually using them.
     
     To illustrate the difference between memory handling with a regular
     dictionary and :class:`WeakValueDictionary`, let's go experiment with
     explicitly calling the garbage collector again:
     
     .. include:: weakref_valuedict.py
         :literal:
         :start-after: #end_pymotw_header
     
     Notice that any loop variables that refer to the values we are caching must be
     cleared explicitly to decrement the reference count on the object. Otherwise
     the garbage collector would not remove the objects and they would remain in
     the cache. Similarly, the all_refs variable is used to hold references to
     prevent them from being garbage collected prematurely.
     
     .. {{{cog
     .. cog.out(run_script(cog.inFile, 'weakref_valuedict.py'))
     .. }}}
     
     ::
     
     	$ python weakref_valuedict.py
     
     	CACHE TYPE: <type 'dict'>
     	all_refs ={'one': ExpensiveObject(one),
     	 'three': ExpensiveObject(three),
     	 'two': ExpensiveObject(two)}
     	Before, cache contains: ['three', 'two', 'one']
     	  three = ExpensiveObject(three)
     	  two = ExpensiveObject(two)
     	  one = ExpensiveObject(one)
     	Cleanup:
     	After, cache contains: ['three', 'two', 'one']
     	  three = ExpensiveObject(three)
     	  two = ExpensiveObject(two)
     	  one = ExpensiveObject(one)
     	demo returning
     	(Deleting ExpensiveObject(three))
     	(Deleting ExpensiveObject(two))
     	(Deleting ExpensiveObject(one))
     	
     	CACHE TYPE: weakref.WeakValueDictionary
     	all_refs ={'one': ExpensiveObject(one),
     	 'three': ExpensiveObject(three),
     	 'two': ExpensiveObject(two)}
     	Before, cache contains: ['three', 'two', 'one']
     	  three = ExpensiveObject(three)
     	  two = ExpensiveObject(two)
     	  one = ExpensiveObject(one)
     	Cleanup:
     	(Deleting ExpensiveObject(three))
     	(Deleting ExpensiveObject(two))
     	(Deleting ExpensiveObject(one))
     	After, cache contains: []
     	demo returning
     
     .. {{{end}}}
     
     The WeakKeyDictionary works similarly but uses weak references for the keys
     instead of the values in the dictionary.
     
     The library documentation for weakref contains this warning:
     
     .. warning::
     
         Caution: Because a WeakValueDictionary is built on top of a Python
         dictionary, it must not change size when iterating over it. This can be
         difficult to ensure for a WeakValueDictionary because actions performed by
         the program during iteration may cause items in the dictionary to vanish
         "by magic" (as a side effect of garbage collection).
     
     .. seealso::
     
         `weakref <http://docs.python.org/lib/module-weakref.html>`_
             Standard library documentation for this module.
     
         :mod:`gc`
             The gc module is the interface to the interpreter's garbage collector.
     

[top] / python / PyMOTW / weakref / index.rst

contact | logmethods.com