Python类方法__init__和__del__构造、析构过程分析


Posted in Python onMarch 06, 2015

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

1、什么时候构造?
2、什么时候析构?
3、成员变量如何处理?
4、Python中的共享成员函数如何访问?
------------------------
探索过程:
1、经过查找,Python中没有专用的构造和析构函数,但是一般可以在__init__和__del__分别完成初始化和删除操作,可用这个替代构造和析构。还有一个__new__用来定制类的创建过程,不过需要一定的配置,此处不做讨论。
2、类的成员函数默认都相当于是public的,但是默认开头为__的为私有变量,虽然是私有,但是我们还可以通过一定的手段访问到,即Python不存在真正的私有变量。如:

__priValue = 0 # 会自动变形为"_类名__priValue"的成员变量

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

测试1:

# encoding:utf8
class NewClass(object):

    num_count = 0 # 所有的实例都共享此变量,即不单独为每个实例分配

    def __init__(self,name):

        self.name = name

        NewClass.num_count += 1

        print name,NewClass.num_count

    def __del__(self):

        NewClass.num_count -= 1

        print "Del",self.name,NewClass.num_count

    def test():

        print "aa"
aa = NewClass("Hello")

bb = NewClass("World")

cc = NewClass("aaaa")
print "Over"

调试运行:

Hello 1

World 2

aaaa 3

Over

DeException l Hello 2

AttributeError: "'NoneType' object has no attribute 'num_count'" in <bound method NewClass.__del__ of <__main__.NewClass object at 0x01AF18D0>> ignored

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的顺序,依次进行,我们无法掌控这里的流程。

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

# encoding:utf8
class NewClass(object):

    num_count = 0 # 所有的实例都共享此变量,即不单独为每个实例分配

    def __init__(self,name):

        self.name = name

        NewClass.num_count += 1

        print name,NewClass.num_count

    def __del__(self):

        NewClass.num_count -= 1

        print "Del",self.name,NewClass.num_count

    def test():

        print "aa"
aa = NewClass("Hello")

bb = NewClass("World")

cc = NewClass("aaaa")
del aa

del bb

del cc
print "Over"

调试输出:

Hello 1

World 2

aaaa 3

Del Hello 2

Del World 1

Del aaaa 0

Over

OK,一切按照我们预料的顺序发生。
但是,我们总不能每次都手动回收吧?这么做Python自己的垃圾回收还有什么意义?

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

# encoding:utf8
class NewClass(object):

    num_count = 0 # 所有的实例都共享此变量,即不单独为每个实例分配

    def __init__(self,name):

        self.name = name

        self.__class__.num_count += 1

        print name,NewClass.num_count

    def __del__(self):

        self.__class__.num_count -= 1

        print "Del",self.name,self.__class__.num_count

    def test():

        print "aa"
aa = NewClass("Hello")

bb = NewClass("World")

cc = NewClass("aaaa")
print "Over"

结果:

Hello 1

World 2

aaaa 3

Over

Del Hello 2

Del World 1

Del aaaa 0

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

PS:

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

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

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

实验中发现垃圾回收自动调用了__del__, 这与书上所说又不符,不知是什么原因,需要继续学习。

Python 相关文章推荐
web.py在模板中输出美元符号的方法
Aug 26 Python
一个基于flask的web应用诞生 记录用户账户登录状态(6)
Apr 11 Python
如何用python整理附件
May 13 Python
python爬虫简单的添加代理进行访问的实现代码
Apr 04 Python
python正则表达式匹配不包含某几个字符的字符串方法
Jul 23 Python
使用OpenCV-python3实现滑动条更新图像的Canny边缘检测功能
Dec 12 Python
使用python turtle画高达
Jan 19 Python
python实现飞机大战游戏(pygame版)
Oct 26 Python
pytorch快速搭建神经网络_Sequential操作
Jun 17 Python
在pytorch中动态调整优化器的学习率方式
Jun 24 Python
如何清空python的变量
Jul 05 Python
python使用pygame创建精灵Sprite
Apr 06 Python
Python列表生成器的循环技巧分享
Mar 06 #Python
Python装饰器使用示例及实际应用例子
Mar 06 #Python
Python迭代器和生成器介绍
Mar 06 #Python
Python __setattr__、 __getattr__、 __delattr__、__call__用法示例
Mar 06 #Python
Python比较文件夹比另一同名文件夹多出的文件并复制出来的方法
Mar 05 #Python
Python挑选文件夹里宽大于300图片的方法
Mar 05 #Python
python基于windows平台锁定键盘输入的方法
Mar 05 #Python
You might like
PHP面向对象之旅:深入理解static变量与方法
2014/01/06 PHP
php中Ctype函数用法详解
2014/12/09 PHP
php实现模拟登陆方正教务系统抓取课表
2015/05/19 PHP
javascript multibox 全选
2009/03/22 Javascript
jQuery 学习第六课 实现一个Ajax的TreeView
2010/05/17 Javascript
jquery.validate分组验证代码
2011/03/17 Javascript
读JavaScript DOM编程艺术笔记
2011/11/15 Javascript
javascript相等运算符与等同运算符详细介绍
2013/11/09 Javascript
jquery ajax对特殊字符进行转义防止js注入使用示例
2013/11/21 Javascript
Jquery获得控件值的三种方法总结
2014/02/13 Javascript
js实现模拟计算器退格键删除文字效果的方法
2015/05/07 Javascript
JS提交form表单实例分析
2015/12/10 Javascript
jQuery Timelinr实现垂直水平时间轴插件(附源码下载)
2016/02/16 Javascript
vue2.0开发实践总结之疑难篇
2016/12/07 Javascript
基于Bootstrap漂亮简洁的CSS3价格表(附源码下载)
2017/02/28 Javascript
js排序与重组的实例讲解
2017/08/28 Javascript
解析Vue.js中的组件
2018/02/02 Javascript
详解VUE自定义组件中用.sync修饰符与v-model的区别
2018/06/26 Javascript
jQuery中实现text()的方法
2019/04/04 jQuery
一次让你了解全部JavaScript的作用域
2019/06/24 Javascript
JavaScript设计模式--简单工厂模式定义与应用案例详解
2020/05/23 Javascript
解决vue-pdf查看pdf文件及打印乱码的问题
2020/11/04 Javascript
[46:42]DOTA2-DPC中国联赛正赛 Aster vs Magma BO3 第二场 3月5日
2021/03/11 DOTA
Python不规范的日期字符串处理类
2014/06/10 Python
Django 连接sql server数据库的方法
2018/06/30 Python
Python中Numpy ndarray的使用详解
2019/05/24 Python
Python之修改图片像素值的方法
2019/07/03 Python
python numpy生成等差数列、等比数列的实例
2020/02/25 Python
基于python实现计算且附带进度条代码实例
2020/03/31 Python
美国儿童运动鞋和服装零售商:Kids Foot Locker
2017/08/05 全球购物
丝芙兰新加坡官网:Sephora新加坡
2018/12/04 全球购物
优秀研究生自我鉴定
2013/12/04 职场文书
读群众路线的心得体会
2014/09/03 职场文书
工程技术员岗位职责
2015/04/11 职场文书
django如何自定义manage.py管理命令
2021/04/27 Python
Python数据类型最全知识总结
2021/05/31 Python