【第041讲心得】【认识__new__构造函数】
本帖最后由 heidern0612 于 2018-12-28 17:53 编辑写心得的过程都是自我思考的过程,借鉴了很多论坛前辈和互联网大佬的经验,仓促间难免有疏漏,如有错误,恳请指出,不胜感激。
一、 回顾__init__方法:
之前学过__init__方法,大概很多同学都很熟悉了,__init__ 方法通常用在初始化一个类实例的时候。
例如:
class Person(object):
"""功夫巨星"""
def __init__(self, name, age):
self.name = name
self.age = age
def __str__(self):
return f'<人物是{self.name},年龄为{self.age}>'
if __name__ == '__main__':
person = Person('Tony Jaa', 24)
print (person)
这样便是__init__最普通的用法了。
但__init__其实不是实例化一个类的时候第一个被调用 的方法。
当使用 Persion(name, age) 这样的表达式来实例化一个类时,最先被调用的方法 其实是 __new__ 方法。
二、__new__ 方法是咩?(原谅我调皮放荡不羁爱自由{:10_297:} )
当你实例化一个对象的时候,首先就会执行__new__ 方法里面的方法。
__new__方法在类定义中不是必须写的,如果没定义,默认会调用object.__new__去创建一个对象。
__new__方法接受的参数虽然也是和__init__一样,但__init__是在类实例创建之后调用,而 __new__方法正是创建这个类实例的方法。
class Person(object):
"""功夫巨星"""
def __new__(cls, name, age):
print ('__new__在实例化之前被调用。')
return object.__new__(cls)
def __init__(self, name, age):
print("__init__在实例化后被调用。")
self.name = name
self.age = age
def __str__(self):
return f'<人物是{self.name},年龄为{self.age}>'
if __name__ == '__main__':
person = Person('Tony Jaa', 24)
print (person)
通过运行这段代码,我们可以看到,__new__方法的调用是发生在__init__之前的。
其实当 你实例化一个类的时候,python具体的执行逻辑是这样的:
1.person = Person(name, age)实例化;
2.首先执行使用name和age参数来执行Person类的__new__方法,这个__new__方法会 返回Person类的一个实例;
*需要注意的是,这里__new__方法必须有一个返回值,返回实例化出来的实例。
可以return父类__new__()出来的实例,也可以直接将object的__new__()出来的实例返回。
3.然后利用这个实例来调用类的__init__方法,上一步里面__new__产生的实例也就是 __init__里面的的self;
所以,__init__ 和 __new__ 最主要的区别在于:
1.__init__ 通常用于初始化一个新实例,控制这个初始化的过程,比如添加一些属性, 做一些额外的操作,发生在类实例被创建完以后。它是实例级别的方法。
2.__new__ 通常用于控制生成一个新实例的过程。它是类级别的方法。
但是说了这么多,__new__最通常的用法是什么呢,我们什么时候需要__new__?
三、__new__ 的作用
依照Python官方文档的说法,__new__方法主要是当你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。
P.S:还有就是实现自定义的metaclass(元类)。
首先我们来看一下第一个功能,具体我们可以用int来作为一个栗子:
假如我们需要一个永远都是正数的整数类型,通过继承int,我们可能会写出这样的代码。
class PositiveInteger(int):
def __init__(self, value):
int.__init__(abs(value))
i = PositiveInteger(-3)
print (i)
但运行后会发现,结果根本不是我们想的那样,我们任然得到了-3。
这是因为对于int这种 不可变的对象,我们只有重载它的__new__方法才能起到自定义的作用。
这是修改后的代码:
class PositiveInteger(int):
def __new__(cls, value):
return int.__new__(cls, abs(value))
i = PositiveInteger(-3)
print (i)
通过重载__new__方法,我们实现了需要的功能。
总结以上:
__new__方法通常用来控制的实例产生的过程,它可以对这个类做一些配置或者处理。
__init__方法主要是用来初始化一些实例的参数,比如添加一些属性,或者做一些其他的事情,这个初始化过程发生在实例被创建以后。
__new__用于对象的创建,是一个静态方法,第一个参数是cls。(想想也是,不可能是self,对象还没创建,哪来的self)
__init__ 用于对象的初始化, 是一个实例方法,第一个参数是self。
先有创建,才有初始化。
即先__new__,而后__init__。
0.类方法中名字前后以__包围的方法名
1.__new__()
2.需要传入参数的时候
3.__init__()方法返回值为None
4.你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径
5.对象被垃圾回收机制启用的时候自动调用
1.class C2F():
def __init__(self,num):
self.num = num
def __repr__(self):
print(self.num*1.8+32)
2.def str2int(value):
num = 0
for i in value:
num += ord(i)
return num
class Nint(int):
def __new__(cls,value):
if isinstance(value,str):
return int.__new__(cls, str2int(value))
else:
return int.__new__(cls,value)
__new__用来做单例用的,楼主说明的很不错。 什么,看不懂 请问为什么调用__new__方法必须要在括号中加入cls参数,而调用实例方法时候,不需要再括号中再加入self参数
页:
[1]