【问题】
__new__和__init__的区别,说法正确的是?
正确答案: A B C D 你的答案: A B C D
A.__new__是一个静态方法,而__init__是一个实例方法
B.__new__方法会返回一个创建的实例,而__init__什么都不返回
C.只有在__new__返回一个cls的实例时,后面的__init__才能被调用
D.当创建一个新实例时调用__new__,初始化一个实例时用__init__
【知识点】
object.__new__(cls[, ...])调用以创建一个 cls 类的新实例。
- __new__() 是一个静态方法 (因为是特例所以你不需要显式地声明),它会将所请求实例所属的类作为第一个参数。其余的参数会被传递给对象构造器表达式 (对类的调用)。
- __new__() 的目的主要是允许不可变类型的子类 (例如 int, str 或 tuple) 定制实例创建过程。它也常会在自定义元类中被重载以便定制类创建过程。
- __new__() 的返回值应为新对象实例 (通常是 cls 的实例)。典型的实现会附带适宜的参数使用 super().__new__(cls[, ...]),通过超类的 __new__() 方法来创建一个类的新实例,然后根据需要修改新创建的实例再将其返回。
- 如果 __new__() 在构造对象期间被发起调用并且它返回了一个实例或 cls 的子类,则新实例的 __init__() 方法将以 __init__(self[, ...]) 的形式被发起调用,其中 self 为新实例而其余的参数与被传给对象构造器的参数相同。
- 如果 __new__() 未返回一个 cls 的实例,则新实例的 __init__() 方法就不会被执行。
object.__init__(self[, ...])在实例 (通过 __new__()) 被创建之后,返回调用者之前调用。其参数与传递给类构造器表达式的参数相同。
- 一个基类如果有 __init__() 方法,则其所派生的类如果也有 __init__() 方法,就必须显式地调用它以确保实例基类部分的正确初始化;例如: super().__init__([args...]).
- 因为对象是由 __new__() 和 __init__() 协作构造完成的 (由 __new__() 创建,并由 __init__() 定制),所以 __init__() 返回的值只能是 None,否则会在运行时引发 TypeError。
【总结】
- __init__ 方法为初始化方法, __new__方法才是真正的构造函数。
- __new__方法默认返回实例对象供__init__方法、实例方法使用。
- __init__ 方法为初始化方法,为类的实例提供一些属性或完成一些动作。
- __new__ 方法创建实例对象供__init__ 方法使用,__init__方法定制实例对象。
- __new__是一个静态方法,而__init__是一个实例方法。