最近在看小甲鱼的python视频,在讲到魔法方法__new__时,用了一段例子
class CapStr(str):
def __new__(cls, string):
string = string.upper()
return str.__new__(cls, string)
感觉不是很明白,查看很多例子加自己测试后作出如下结论:
首先要明确的是__new__方法一定要返回一个实例对象,所以这也就是为什么不能直接 return string 的原因,因为如果
直接return string,我们可以做个测试:
这里就有问题了,我们执行了 a = CapStr('i love china')之后,a不应该是个CapStr类型的对象吗,为什么type是‘str’呢,原因就是因为我们返回的是return string。所以 __new__方法一定要返回一个实例对象的,只有这样,才能执行后面的__init__方法。
那么还有一个问题,为什么要返回 return str.__new__(cls, string)呢,为什么要这么写呢?我查了网上有人说了这么一句话,只有调用父类的__new__方法,返回的才是当前类的实例对象,对此种说法我有点怀疑,并自己通过测试代码得到如下结论:
class A:
pass
class B:
def __new__(cls):
return A.__new__(cls)
代码中A和B没有任何的继承关系,那我们运行得出什么结论呢?如下图:
结果可知,c的类型仍然是B,所以感觉return 中返回的仅仅是借用某一个类(A)中的__new__方法实现了返回当前参数类型的实例对象而已,并不会在意是否是其父类、调用哪个类的__new__方法。
最后声明,__new__(cls)方法,需要的cls参数一定是当前的类,此处的cls可以是任何字符,返回的一定是个实例对象,才会执行后面的__init__方法。