我们通常把 __init__
称为构造方法,这是从其他语言借鉴过来的术语。其实,用于构建实 例的是特殊方法 __new__
:这是个类方法(使用特殊方式处理,因此不必使用 @classmethod
装饰器),必须返回一个实例
。返回的实例会作为第一个参数(即 self
)传给 __init__
方 法。因为调用 __init__
方法时要传入实例,而且禁止返回任何值,所以 __init__
方法其实是“初始化方法”。真正的构造方法是__new__
。我们几乎不需要自己编写 __new__
方法, 因为从 object
类继承的实现已经足够了。
刚才说明的过程,即从__new__
方法到__init__
方法,是最常见的,但不是唯一的。 __new__
方法也可以返回其他类的实例,此时,解释器不会调用 __init__
方法。
也就是说,Python 构建对象的过程可以使用下述伪代码概括:
# 构建对象的伪代码
def object_maker(the_class, some_arg):
new_object = the_class.__new__(some_arg)
if isinstance(new_object, the_class):
the_class.__init__(new_object, some_arg)
return new_object
# 下述两个语句的作用基本等效
x = Foo('bar')
x = object_maker(Foo, 'bar')
在看一个示例:
from collections import abc
class FrozenJSON:
"""
一个只读接口,使用属性表示法访问JSON类对象
"""
def __new__(cls, arg):
if isinstance(arg, abc.Mapping):
return super().__new__(cls)
elif isinstance(arg, abc.MutableSequence):
return [cls(item) for item in arg]
else:
return arg
def __init__(self, mapping):
self.__data = {}
for key, value in mapping.items():
if iskeyword(key)
key += '_'
self.__data[key] = value