python_如何在循环引用中管理内存?

时间:2024-01-08 19:39:32

案例:

  python中通过引用计数来回收垃圾对象,在某些环形数据结构(树,图……),存在对象间的循环引用,比如树的父节点引用子节点,子节点同时引用父节点,此时通过del掉引用父子节点,两个对象不能被立即释放

需求:

如何解决此类的内存管理问题?

如何查询一个对象的引用计数?

import sys

sys.getrefcount(obj)

# 查询引用计数必多 1 ,因为object也引用 查询对象

如何解决内存管理问题?

  1. 通过weakref,进行弱引用,当del时候,不再引用,在引用方添加weakref.ref(引用obj),
  2. 使用引用的时候,需要用到函数调用的形式
    #!/usr/bin/python3
    
    import weakref
    import sys
    
    class Data(object):
        def __init__(self, value, owner):
            self.value = value
    
            # 声明弱引用,owner为Node类本身
            self.owner = weakref.ref(owner)
    
        # 通过函数调用的方式访问引用对象
        def __str__(self):
            return "%s's data, value is %s" % (self.owner(), self.value)
    
        def __del__(self):
            print('in_data.__del__')
    
    class Node(object):
        def __init__(self, value):
    
            # 把类本身,也当做参数传入Data类中
            self.data = Data(value, self)
    
        # 自定义对象名,容易辨认
        def __str__(self):
            return 'Node'
    
        def __del__(self):
            print('in_node.__del__')
    
    if __name__ == '__main__':
        node = Node(100)
        print(node.data)
    
        # 打印node对象的引用计数
        print(sys.getrefcount(node) - 1)
    
        # 当删除node对象时候,Data实例对象在引用计算为0也相应释放
        del node
    
        input('del done >>>>>')