Python中的__init__和__new__介绍

介绍

首先我们要知道在面向对象编程中,实例化基本遵循创建实例对象、初始化实例对象、最后返回实例对象这么一个过程。

Python 中的 __new__ 方法负责创建一个实例对象,__init__ 方法负责将该实例对象进行初始化。

__new__() 是在新式类中新出现的方法,它作用在构造方法建造实例之前,可以这么理解,在 Python 中存在于类里面的构造方法 __init__() 负责将类的实例化,而在 __init__() 启动之前,__new__() 决定是否要使用该 __init__() 方法,因为__new__() 可以调用其他类的构造方法或者直接返回别的对象来作为本类的实例。

__new__(),第一个参数cls是当前正在实例化的类。

class Example(object):

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

重点:

事实上如果(新式)类中没有重写__new__()方法,即在定义新式类时没有重新定义__new__()时,Python默认是调用该类的直接父类的__new__()方法来构造该类的实例,如果该类的父类也没有重写__new__(),那么将一直按此规矩追溯至object的__new__()方法,因为object是所有新式类的基类。

而如果新式类中重写了__new__()方法,那么你可以*选择任意一个的其他的新式类(必定要是新式类,只有新式类必定都有__new__(),因为所有新式类都是object的后代,而经典类则没有__new__()方法)的__new__()方法来制造实例,包括这个新式类的所有前代类和后代类,只要它们不会造成递归死循环。

如果__new__()没有返回cls(即当前类)的实例,那么当前类的__init__()方法是不会被调用的。如果__new__()返回其他类(新式类或经典类均可)的实例,那么只会调用被返回的那个类的构造方法。

__new__ 的作用

依照Python官方文档的说法,__new__方法主要是当你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。还有就是实现自定义的metaclass。

首先我们来看一下第一个功能,具体我们可以用int来作为一个例子:

假如我们需要一个永远都是正数的整数类型,通过集成int,我们可能会写出这样的代码。

class ExampleInteger(int):
def __init__(self, value):
super(ExampleInteger, self).__init__(self, abs(value)) i = ExampleInteger(-3)
print i

但运行后会发现,结果根本不是我们想的那样,我们任然得到了-3。这是因为对于int这种 不可变的对象,我们只有重载它的__new__方法才能起到自定义的作用。

这是修改后的代码:

写法一:

class ExampleInteger(int):

    def __new__(cls, value):
return int.__new__(cls, abs(value)) i = ExampleInteger(-3)
print i

写法二:

class ExampleInteger(int):

    def __new__(cls, value):
return super(ExampleInteger, cls).__new__(cls, abs(value)) i = ExampleInteger(-3)
print i

通过重载__new__方法,我们实现了需要的功能。

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

super

super 是用来解决多重继承问题的,直接用类名调用父类方法在使用单继承的时候没问题,但是如果使用多继承,会涉及到查找顺序(MRO)、重复调用(钻石继承)等种种问题。

参考文章: http://blog.csdn.net/johnsonguo/article/details/585193

上一篇:Python中的lambda的简单介绍


下一篇:(数据科学学习手札32)Python中re模块的详细介绍