Python 的 __new__()方法与实例化

__new__() 是新式类中才有的方法,它执行在构造方法建立实例以前。能够这么理解,在 Python 中类中的构造方法 __init__() 负责将类实例化,而在 __init__() 启动以前,__new__() 决定是否要使用该 __init__() 方法(由于__new__() 能够调用其余类的构造方法或者直接返回别的对象来做为本类的实例)。html

若是将类比喻成工厂,那么__init__()方法则是该工厂的生产工人,__init__()方法接受的初始化参数则是生产所需的原料,__init__()方法会按照方法中的语句负责将原料加工成实例以供工厂出货。而__new__()则是生产部经理,__new__()方法既能够将原料提供给该生产部门的工人,也能够将原料提供给其余的外来单位,由于这名经理能够借该工厂的名义向其余单位出让生产原料(假公济私)。python

__new__() 方法的特性:安全

  • __new__() 方法是在类准备将自身实例化时调用。
  • __new__() 方法始终都是类的静态方法,即便没有被加上静态方法装饰器。
  • 类的实例化和它的构造方法一般都是这个样子:
class MyClass(object):
    def __init__(self, *args, **kwargs):
        ...

# 实例化
myclass = MyClass(*args, **kwargs)

正如以上所示,一个类能够有多个位置参数和多个命名参数,而在实例化开始以后,在调用 __init__() 方法以前,Python 首先调用 __new__() 方法:框架

def __new__(cls, *args, **kwargs):
    ...

第一个参数cls是当前正在实例化的类。spa

  • 若是要获得当前类的实例,应当在当前类中的 __new__() 方法语句中调用当前类的父类的 __new__() 方法。

  例如,若是当前类是直接继承自 object,那当前类的 __new__() 方法返回的对象应该为:设计

def __new__(cls, *args, **kwargs):
    ...
    return object.__new__(cls)

注意:htm

  事实上若是(新式)类中没有重写__new__()方法,即在定义新式类时没有从新定义__new__()时,Python默认会调用该类的直接父类的__new__()方法来构造该类的实例,若是该类的父类也没有重写__new__(),那么将一直按此规矩追溯直至object的__new__()方法,由于object是全部新式类的基类。 对象

  若是新式类中重写了__new__()方法,那么你能够选择任意一个该新式类的全部前代类和后代类((只能是新式类,经典类中没有__new__()方法)的__new__()方法来制造实例),只要它们不会形成递归死循环。具体看如下代码解释:blog

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

# 以上return等同于 
# return object.__new__(Foo, *args, **kwargs)
# return Stranger.__new__(cls, *args, **kwargs)
# return Child.__new__(cls, *args, **kwargs)

class Child(Foo):
    def __new__(cls, *args, **kwargs):
        return object.__new__(cls, *args, **kwargs)

# 若是Child中没有定义__new__()方法,那么会自动调用其父类的__new__()方法来制造实例,即 Foo.__new__(cls, *args, **kwargs)。
# 在任何新式类的__new__()方法,不能调用自身的__new__()来制造实例,由于这会形成死循环。所以必须避免相似如下的写法:
# 在Foo中避免:return Foo.__new__(cls, *args, **kwargs)或return cls.__new__(cls, *args, **kwargs)。Child同理。
# 使用object或者没有血缘关系的新式类的__new__()是安全的,可是若是是在有继承关系的两个类之间,应避免互调形成死循环,
# 例如:(Foo)return Child.__new__(cls), (Child)return Foo.__new__(cls)。

class Stranger(object):
    ...
# 在制造Stranger实例时,会自动调用 object.__new__(cls)
  • 一般来讲,新式类开始实例化时,__new__()方法会返回cls(cls指代当前类)的实例,而后该类的__init__()方法做为构造方法会接收这个实例(即self)做为本身的第一个参数,而后依次传入__new__()方法中接收的位置参数和命名参数。

注意:若是__new__()没有返回cls(即当前类)的实例,那么当前类的__init__()方法是不会被调用的。若是__new__()返回其余类(新式类或经典类都可)的实例,那么只会调用被返回的那个类的构造方法。继承

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

class Stranger(object):
    ...

foo = Foo()
print type(foo)    

# 打印的结果显示foo实际上是Stranger类的实例。

# 所以能够这么描述__new__()和__init__()的区别,在新式类中__new__()才是真正的实例化方法,
# 为类提供外壳制造出实例框架,而后调用该框架内的构造方法__init__()使其丰满。
# 若是以建房子作比喻,__new__()方法负责开发地皮,打下地基,并将原料存放在工地。而__init__()方法
# 负责从工地取材料建造出地皮开发招标书中规定的大楼,__init__()负责大楼的细节设计,建造,装修使其可交付给客户。

参考了: http://www.cnblogs.com/ifantastic/p/3175735.html

相关文章
相关标签/搜索