python __del__

简介:

最近学习《Python参考手册》学到Class部分,遇到了类的构造析构部分的问题:

1、什么时候构造?

2、什么时候析构?

3、成员变量如何处理?

4、Python中的共享成员函数如何访问?

------------------------

探索过程:

1、经过查找,Python中没有专用的构造和析构函数,但是一般可以在__init__和__del__分别完成初始化和删除操作,可用这个替代构造和析构。还有一个__new__用来定制类的创建过程,不过需要一定的配置,此处不做讨论。 

2、类的成员函数默认都相当于是public的,但是默认开头为__的为私有变量,虽然是私有,但是我们还可以通过一定的手段访问到,即Python不存在真正的私有变量。如:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. __priValue = # 会自动变形为"_类名__priValue"的成员变量  


3、由于Python的特殊性,全局成员变量是共享的,所以类的实例不会为它专门分配内容空间,类似于static,具体使用参看下面的例子。

测试1:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. # encoding:utf8  
  2.   
  3. class NewClass(object):  
  4.     num_count = # 所有的实例都共享此变量,即不单独为每个实例分配  
  5.     def __init__(self,name):  
  6.         self.name = name  
  7.         NewClass.num_count += 1  
  8.         print name,NewClass.num_count  
  9.     def __del__(self):  
  10.         NewClass.num_count -= 1  
  11.         print "Del",self.name,NewClass.num_count  
  12.     def test():  
  13.         print "aa"  
  14.   
  15. aa = NewClass("Hello")  
  16. bb = NewClass("World")  
  17. cc = NewClass("aaaa")  
  18.   
  19. print "Over"  

调试运行:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. Hello 1  
  2. World 2  
  3. aaaa 3  
  4. Over  
  5. DeException l Hello 2  
  6. AttributeError: "'NoneType' object has no attribute 'num_count'" in <bound method NewClass.__del__ of <__main__.NewClass object at 0x01AF18D0>> ignored  
  7. Exception AttributeError: "'NoneType' object has no attribute 'num_count'" in <bound method NewClass.__del__ of <__main__.NewClass object at 0x01AF1970>> ignored  

       我们发现,num_count 是全局的,当每创建一个实例,__init__()被调用,num_count 的值增一,当程序结束后,所有的实例会被析构,即调用__del__() 但是此时引发了异常。查看异常为 “NoneType” 即 析构时NewClass 已经被垃圾回收,所以会产生这样的异常。

        但是,疑问来了?为什么会这样?按照C/C++等语言的经验,不应该这样啊!经过查找资料,发现:

        Python的垃圾回收过程与常用语言的不一样,Python按照字典顺序进行垃圾回收,而不是按照创建顺序进行。所以当系统进行回收资源时,会按照类名A-Za-z的顺序,依次进行,我们无法掌控这里的流程。

     明白这些,我们做如下尝试:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. # encoding:utf8  
  2.   
  3. class NewClass(object):  
  4.     num_count = # 所有的实例都共享此变量,即不单独为每个实例分配  
  5.     def __init__(self,name):  
  6.         self.name = name  
  7.         NewClass.num_count += 1  
  8.         print name,NewClass.num_count  
  9.     def __del__(self):  
  10.         NewClass.num_count -= 1  
  11.         print "Del",self.name,NewClass.num_count  
  12.     def test():  
  13.         print "aa"  
  14.   
  15. aa = NewClass("Hello")  
  16. bb = NewClass("World")  
  17. cc = NewClass("aaaa")  
  18.   
  19. del aa  
  20. del bb  
  21. del cc  
  22.   
  23. print "Over"  


调试输出:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. Hello 1  
  2. World 2  
  3. aaaa 3  
  4. Del Hello 2  
  5. Del World 1  
  6. Del aaaa 0  
  7. Over  

OK,一切按照我们预料的顺序发生。

但是,我们总不能每次都手动回收吧?这么做Python自己的垃圾回收还有什么意义?

SO,继续查找,我们还可以通过self.__class__访问到类本身,然后再访问自身的共享成员变量,即 self.__class__.num_count , 将类中的NewClass.num_count替换为self.__class__.num_count 编译运行,如下:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. # encoding:utf8  
  2.   
  3. class NewClass(object):  
  4.     num_count = # 所有的实例都共享此变量,即不单独为每个实例分配  
  5.     def __init__(self,name):  
  6.         self.name = name  
  7.         self.__class__.num_count += 1  
  8.         print name,NewClass.num_count  
  9.     def __del__(self):  
  10.         self.__class__.num_count -= 1  
  11.         print "Del",self.name,self.__class__.num_count  
  12.     def test():  
  13.         print "aa"  
  14.   
  15. aa = NewClass("Hello")  
  16. bb = NewClass("World")  
  17. cc = NewClass("aaaa")  
  18.   
  19. print "Over"  

结果:

[python]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. Hello 1  
  2. World 2  
  3. aaaa 3  
  4. Over  
  5. Del Hello 2  
  6. Del World 1  
  7. Del aaaa 0  

Perfect!我们完美地处理了这个问题!

 

PS:

书上又提到了一些问题,在这里作补充(仅作为参考):

__new__()是唯一在实例创建之前执行的方法,一般用在定义元类时使用。

del xxx 不会主动调用__del__方法,只有引用计数==0时,__del__()才会被执行,并且定义了__del_()的实例无法被Python的循环垃圾收集器收集,所以尽量不要自定义__del__()。一般情况下,__del__() 不会破坏垃圾处理器。


本文转自cococo点点博客园博客,原文链接:http://www.cnblogs.com/coder2012/p/4050084.html,如需转载请自行联系原作者

相关文章
|
Python
49 python - __del__()方法
49 python - __del__()方法
83 0
|
Python
python之列表中常用的函数:append,extend,insert,pop,remove,del函数的定义与使用方法,元素是否在列表中的判断
python之列表中常用的函数:append,extend,insert,pop,remove,del函数的定义与使用方法,元素是否在列表中的判断
360 0
|
开发者 索引 Python
【Python 基础】remove、del和pop有什么区别?
【5月更文挑战第8天】【Python 基础】remove、del和pop有什么区别?
|
Python
python 中__init__ ,__new__ ,__call__,__del__ 方法
python 中__init__ ,__new__ ,__call__,__del__ 方法
294 0
python--__repr__、__del__、__eq__
python--__repr__、__del__、__eq__
|
Java C++ Python
【Python学习笔记】Python中的del
【Python学习笔记】Python中的del
|
存储 Java 索引
21.从入门到精通:Python数据结构 列表 将列表当做堆栈使用 将列表当作队列使用 列表推导式 嵌套列表解析 del 语句
21.从入门到精通:Python数据结构 列表 将列表当做堆栈使用 将列表当作队列使用 列表推导式 嵌套列表解析 del 语句
|
Python
Python面向对象、类的抽象、类的定义、类名遵循大驼峰的命名规范创建对象、类外部添加和获取对象属性、类内部操作属性魔法方法__init__()__str__()__del__()__repr__()
面向对象和面向过程,是两种编程思想. 编程思想是指对待同一个问题,解决问题的套路方式.面向过程: 注重的过程,实现的细节.亲力亲为.面向对象: 关注的是结果, 偷懒.类和对象,是面向对象中非常重要的两个概念object 是所有的类基类,即最初始的类class 类名(object): 类中的代码PEP8代码规范:类定义的前后,需要两个空行 创建的对象地址值都不一样如dog和dog1的地址就不一样,dog的地址为2378043254528dog1的地址为2378044849840 8.类内部操作属性 sel
387 1
Python面向对象、类的抽象、类的定义、类名遵循大驼峰的命名规范创建对象、类外部添加和获取对象属性、类内部操作属性魔法方法__init__()__str__()__del__()__repr__()
|
Java Python
Python - 面向对象编程 - __del__() 析构方法
Python - 面向对象编程 - __del__() 析构方法
196 0
|
Python
Python----魔法函数__init__/__del__/__new__的用法
Python----魔法函数__init__/__del__/__new__的用法
141 0

推荐镜像

更多