new()方法是在新式类(新式类和经典类的区别)中的方法。object为所有新式类的基类,在object中,new()方法被定义为静态方法,并且至少需要传递一个参数cls,cls表示需要实例化的类。
在创建一个类对象实例的过程中,new()方法作用在构造方法__init__()之前。new()函数执行后会返回实例对象(self),然后将self作为第一个参数传给该类的初始化方法__init__()方法。
可以这么理解,如果设计一个A类,A类中__new__()方法负责返回一个实例对象(不一定是A类对象),而__init__()方法负责将类对象初始化(设置各种属性之类)。在init()调用之前,new()可以决定是否要使用init()方法,因为new()可以调用其他类的构造方法或者直接返回别的对象来作为本类(A类)的实例。
下面来看一个例子:
#类A
class A(object):
def __init__(self):
self.name = 'A' #设置name属性
def __new__(cls, *args, **kwargs):
# 执行object的__new__()函数,传入cls说明返回当前类(A类)的实例对象
return object.__new__(cls)
#类B
class B(object):
def __init__(self):
self.name = 'B' #设置name属性
def __new__(cls, *args, **kwargs):
#执行object的__new__()函数,传入参数A说明返回(A类)的实例对象
return object.__new__(A)
a = A() #创建A类实例, 相当于隐式执行了A类的__new__()和__init__()
b = B() #创建B类实例, 只执行了B类的__new__()
print(a.name) # A
#print(b.name) # 出错:AttributeError: 'A' object has no attribute 'name'
b.__init__( ) #显示调用__init(),完成b的初始化
print(b.name) # A 正常输出
print(type(a)) # <class '__main__.A'>
print(type(b)) # <class '__main__.A'>
#第一个print(b.name)报错的原因是,想要创建B类实例的时候,new方法却直接返回了A类实例对象
#因此,没有执行B类的init方法,也没有执行A类的init方法,也就导致b对象没有name属性
#在显示调用init方法之后,b才有了name属性,所以第二个的print(b.name)正常输出