__init__()有一个参数self,该self参数就是__new__()返回的实例(原来__new__()返回的是实例.)

 

大家可能对Python中的__init__方法很熟悉,认为他是实例化类时调用的第一个方法。但其实他并不是。实例化时调用的第一个方法其实是__new__方法。

  1  当我们实例化A类对象时,Python中首先调用的是该A类对象的__new__方法,如果该A类对象没有定义__new__方法,则去父类中依次查找,直到object类

     2  object类有一个__new__方法,该方法接收一个参数(一般为类对象),将该参数进行实例化并返回一个对象

  3  Python解释器会将调用__new__方法并将A类对象作为第一个参数传入,最后会返回一个对象(这个对象就是A类的实例对象,我们称之为a1)

  4  Python解释器默认会调用a1对象的__init__方法,并将参数传入。

 

 

object将__new__()方法定义为静态方法,并且至少需要传递一个参数cls,cls表示需要实例化的类,此参数在实例化时由Python解释器自动提供。

 



我们来看下下面类中对__new__()方法的实现:

class Demo(object):
  def __init__(self):
    print '__init__() called...'
 
  def __new__(cls, *args, **kwargs):
    print '__new__() - {cls}'.format(cls=cls)
    return object.__new__(cls, *args, **kwargs)
 
if __name__ == '__main__':
  de = Demo()



发现实例化对象的时候,调用__init__()初始化之前,先调用了__new__()方法

__new__()必须要有返回值,返回实例化出来的实例,需要注意的是,可以return父类__new__()出来的实例,也可以直接将object的__new__()出来的实例返回。

__init__()有一个参数self,该self参数就是__new__()返回的实例,__init__()在__new__()的基础上可以完成一些其它初始化的动作,__init__()不需要返回值。

若__new__()没有正确返回当前类cls的实例,那__init__()将不会被调用,即使是父类的实例也不行。(???)

我们可以将类比作制造商,__new__()方法就是前期的原材料购买环节,__init__()方法就是在有原材料的基础上,加工,初始化商品环节。

 



再来个单例的,通过重载__new__实现单例:

class Singleton(object):
    def __new__(cls, *args, **kwargs): #一般的的类都是self必写。而__new__写的是cls.当运行到这个方法的时候,类如果还没实例化,就是cls,否则是self。
if not hasattr(cls, '_instance'):
            cls._instance = super(Singleton, cls).__new__(cls, *args, **kwargs)
      return cls._instance



 

__new__()方法就是前期的原材料购买环节:举一个购买别的类的例子.



class Foo(object):
    def __init__(self, *args, **kwargs):
        pass
    def __new__(cls, *args, **kwargs):
        return object.__new__(Stranger, *args, **kwargs)

class Stranger(object):
    pass

foo = Foo()
print(type(foo)) #<class '__main__.Stranger'>