【同】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