__new__ AND __init__
构造方法=创建对象+初始化对象=__new__+__init__
__new__方法是在实例创建之前被调用,是一个静态方法,主要的功能就是创建一个类的实例并返回
__init__方法是在实例创建之后被调用,主要的功能是设置实例的一些属性初始值
运行过程:__new__在__init__之前被调用,__new__的返回值(实例)将传递给__init__方法的第一个参数(self),然后__init__给这个实例(self)设置一些参数
In [1]: class Test(object):
...: def __init__(self): # __init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其它初始化的动作,__init__不需要返回值
...: print(self)
...: print('这是init方法')
...: def __new__(cls):
...: print(id(cls))
...: print('这是new方法')
...: ret = object.__new__(cls) # __new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供,所有类的默认父类都是object
...: print(ret)
...: return ret # __new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例
...:
In [2]: print(id(Test)) #返回对象的内存地址
2618556841832
In [3]: test = Test()
2618556841832
这是new方法
<__main__.Test object at 0x00000261AFAFBC48>
<__main__.Test object at 0x00000261AFAFBC48>
这是init方法
相当于做了三件事:
- 调用new来创建对象,找了一个变量ret接受new的返回值,这个值表示创建对象的引用
- init(刚刚创建出来的对象的引用),设置一些初始值
- 返回对象的引用
__call__
call的作用是使实例能够像一个函数一样被调用,同时不影响实例本身的生命周期(即不影响实例的构造和析构),但call方法可以改变实例内部变量的值
class X(object):
def __init__(self, a, b, range):
self.a = a
self.b = b
self.range = range
def __call__(self, a, b):
self.a = a
self.b = b
print('__call__ with ({}, {})'.format(self.a, self.b))
def __del__(self, a, b, range):
del self.a
del self.b
del self.range
#实例化这个类
x_instance = X(1, 2, 3)
#通过_call_()来修改类属性值
x_instance(1,2) # 将实例x看做了一个函数
得到结果:__call__ with (1, 5)