Python __new__和__init__的区别【多测师】

Python 中的__new__和__init__的区别

【同】html

  两者均是Python面向对象语言中的函数,__new__比较少用,__init__则用的比较多。函数

【异】post

  • __new__是在实例建立以前被调用的,由于它的任务就是建立实例而后返回该实例对象,是个静态方法。
  • __init__是当实例对象建立完成后被调用的,而后设置对象属性的一些初始值,一般用在初始化一个类实例的时候。是一个实例方法。

也就是: __new__先被调用,__init__后被调用,__new__的返回值(实例)将传递给__init__方法的第一个参数,而后__init__给这个实例设置一些参数。url

  ===》》》.net

 

 【一些说明】3d

一、继承自object的新式类才有__new__code

二、__new__至少要有一个参数cls,表明当前类,此参数在实例化时由Python解释器自动识别htm

三、__new__必需要有返回值,返回实例化出来的实例,这点在本身实现__new__时要特别注意,能够return父类(经过super(当前类名, cls))__new__出来的实例,或者直接是object的__new__出来的实例对象

四、__init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上能够完成一些其它初始化的动做,__init__不须要返回值blog

五、若是__new__建立的是当前类的实例,会自动调用__init__函数,经过return语句里面调用的__new__函数的第一个参数是 cls 来保证是当前类实例,若是是其余类的类名,;那么实际建立返回的就是其余类的实例,其实就不会调用当前类的__init__函数,也不会调用其余类的__init__函数。

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

七、而若是子类中重写了__new__()方法,那么你能够自由选择任意一个的其余的新式类(一定要是新式类,只有新式类一定都有__new__(),由于全部新式类都是object的后代,而经典类则没有__new__()方法)的__new__()方法来制造实例,包括这个新式类的全部前代类和后代类,只要它们不会形成递归死循环。反正确定不能调用本身的__new__,这确定是死循环。

八、对于子类的__init__,其调用规则跟__new__是一致的,固然若是子类和父类的__init__函数都想调用,能够在子类的__init__函数中加入对父类__init__函数的调用。

九、咱们在使用时,尽可能使用__init__函数,不要去自定义__new__函数,由于这二者在继承派生时的特性仍是很不同的。

十、将类比做制造商,__new__方法就是前期的原材料购买环节,__init__方法就是在有原材料的基础上,加工,初始化商品环节

 

【__init__方法】

 【__new__方法】

__new__方法接受的参数虽然也是和__init__同样,但__init__是在类实例建立以后调用,而 __new__方法正是建立这个类实例的方法。

  ===》

具体的执行逻辑:

1. p = Person(name, age)
2. 首先执行使用name和age参数来执行Person类的__new__方法,这个__new__方法会返回Person类的一个实例(一般状况下是使用 super(Persion, cls).__new__(cls, ... ...) 这样的方式),
3. 而后利用这个实例来调用类的__init__方法,上一步里面__new__产生的实例也就是 __init__里面的的 self。

 因此,__init__ 和 __new__ 最主要的区别在于:
1.__init__ 一般用于初始化一个新实例,控制这个初始化的过程,好比添加一些属性, 作一些额外的操做,发生在类实例被建立完之后。它是实例级别的方法。
2.__new__ 一般用于控制生成一个新实例的过程。它是类级别的方法。

 

【__new__的做用】

依照Python官方文档的说法,__new__方法主要是当你继承一些不可变的class时(好比int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。还有就是实现自定义的metaclass。
首先咱们来看一下第一个功能,具体咱们能够用int来做为一个例子:
假如咱们须要一个永远都是正数的整数类型,经过集成 int,咱们可能会写出这样的代码。

 

但运行后会发现,结果根本不是咱们想的那样,咱们任然获得了-3。这是由于对于int这种 不可变的对象,咱们只有重载它的__new__方法才能起到自定义的做用。
这是修改后的代码:

经过重载__new__方法,咱们实现了须要的功能。

 

 

 

【牛客网上一道题】

 ※  __new__和__init__的区别,说法正确的是? (ABCD

A. __new__是一个静态方法,而__init__是一个实例方法
B. __new__方法会返回一个建立的实例,而__init__什么都不返回
C. 只有在__new__返回一个cls的实例时,后面的__init__才能被调用
D. 当建立一个新实例时调用__new__,初始化一个实例时用__init__



【Reference】
一、https://my.oschina.net/kinegratii/blog/334968
二、https://www.cnblogs.com/suntp/p/6445286.html
三、https://my.oschina.net/liuyuantao/blog/747164
相关文章
相关标签/搜索