?
当前位置:??编程语言>python

Python类的构造函数,析构函数以及垃圾回收机制详细介绍及代码举例

?
分享到:
????发布时间:2014-9-10??


????本文导语:?Python类的构造函数,析构函数以及垃圾回收机制详细介绍及代码举例Python类的构造函数是__init__,它与其它函数的区别,首先是通过前面和后面带两个双下划线来跟别的函数名称进行区分。构造函数: __init__( self )析构函数: &nb...

Python构造函数,析构函数以及垃圾回收机制详细介绍及代码举例

Python类的构造函数是__init__,它与其它函数的区别,首先是通过前面和后面带两个双下划线来跟别的函数名称进行区分。

构造函数: ? __init__( self )

析构函数: ?__del__( self )??

在python中这构造函数和析构函数可以省略。先看简单的例子:

class?Simple:
?????def?__init__(?self?):
?????????pass
?????def?__del__(?self?):
?????????pass
?????def?func(?self?):
?????????pass


上面定义了一个简单的类,一个构造函数,析构函数和普通的函数。

如果觉得不需要Python类的构造函数和析构函数,那么这个类可以简化成

class?Simple:
?????def?func(?self?):
?????????pass


Python类的构造函数在类的构造的时候调用,但是不是一定会调用,这点我会在后面提到原因;析构函数被python的垃圾回收销毁的时候调用。

下面写的简单的类测试一下:

类的定义:

class?Simple:
????def?__init__(?self?):
????????print(?"constructor?called,?id={0}".format(?id(?self?)))
????def?__del__(?self?):
????????print(?"destructor?called,?id={0}".format(?id(?self?)))
????def?func(?sef?):
????????print(?"Simple?func"?)


使用类:

a?=?Simple()?????????#输出->constructor?called,?id?=?3211123
b?=?Simple()?????????#输出->constructor?called,?id?=?3211145
a?=?Simple()?????????#输出->constructor?called,?id?=?32111225??#destructor?called,?id?=?3211123

上面构造2个对象a, b。 我们看到每一次构造一个对象,构造函数就会被调用一次。在构造函数中输出了被构造对象的id(id相当于c/c++对象的地址),是为了和后面的析构函数id比较。

a的id为 ? 3211123

b的id为 ? 3211145

后面为什么又实例化 a呢,因为我们想要利用python的垃圾回收器自动回收我们的之前实例化的a对象,这样先会调用构造函数__init__来构造一个新的对象,这个新的对象的id为32111225,和我们之前a的id不一样。接下来垃圾回收器,发现已经没有变量引用它了,因此就会调用a之前引用的对象id = 3211123的析构函数,然后从内存中彻底销毁它。

注意是先构造新对象,然后才销毁旧对象。

上面我们提到构造函数不是一定调用,那是什么情况下回出现这种情况呢? 如果类提供了__new__这个函数,并且没有返回实例,也就是没有返回解释器传给它的参数,那么构造函数__init__就不会被调用。?


关于Python类的__new__的简单介绍

? ?它的引进是为了让程序员控制实例的创建过程,python的设计人员建议这个方法主要用在子类化稳定类型,如int, str。并且这个方法只有python新风格类(或者新类)才会被调用,新类是以object为基类的类。


class?Old:
???def?__new__(?self?):
?????????return?self
class?New(?object?):
???def?__new__(?self?):
?????????return?super(?New,?self?).__new__(?self?)


上面的New的就是一个新类,因为它继承自object,它的__new__会被调用,而Old类的__new__不会被调用。?

接下来我们做个测试来验证上面所说的。

class?Simple(?object?):
????def?__init__(?self?):
????????print(?"constructor?called,?id={0}".format(?id(?self?)))
????def?__del__(?self?):
????????print(?"destructor?called,?id={0}".format(?id(?self?)))
????def?__new__(?self?):
????????print(?"new?called,?id={0}".format(?id(?self?)))
????????#return?super(?CA,?self?).__new__(?self?)


上面的__new__中我们屏蔽掉了返回实例的一行。然后我们实例化一个对象

a = Simple()

发现什么了,__init__没有被调用吧。??


上面对Python类的构造函数和析构函数做了测试,也知道它的作用和怎么用,下面对它们补充。?


构造函数(__init__) 这个函数在用代码生成一个对象的实例的时候就会被调用

如果基类和子类都有__init__,那么子类必须显式的调用的基类的__init__。?

析构函数 ( __del__) 这个函数一般很难知道在哪里会被调用,如果要指明他的调用的话就要用del语句

如果基类和子类都有__del__,那么子类必须显式的调用的基类的__del__。


Python垃圾回收机制

Python 采用垃圾回收机制来清理不再使用的对象;Python 提供gc模块释放

不再使用的对象,Python 采用‘引用计数’ 的算法方式来处理回收,

即:当某个对象在其作用域内不再被其他对象引用的时候,Python 就自动清除对象;

Python 的函数collect()可以一次性收集所有待处理的对象(gc.collect())


程序举例1:

e9944080_12f9734b4c8g215.jpg


程序举例2:

class?Foo:
????def?__init__(self):
????????print('__init__')
????def?__del__(self):
????????print('__del__')
????def?foo(self):
????????print('foo')
????????
test?=?Foo()
test.foo()


相关文章推荐:


站内导航:


特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

?2012-2019,169IT.COM,E-mail:www_169it_com#163.com(请将#改为@)

浙ICP备11055608号