二问:Py 第43课,算数2,反运算
本帖最后由 Peteryo01223 于 2021-3-15 10:34 编辑Fish C,在‘第43课,算数2’的视频中,第5至6分钟,Fish C 举的反运算例子:
class Nint(int):
def __radd__(self, other):
return int.__sub__(self, other)
run后结果:
>>> a = Nint(5)
>>> b = Nint(3)
>>> a + b # 没有触发反运算 __radd__ 方法
8
>>> 1 + b # 触发了反运算 __radd__ 方法。这里 self 是变量 b, other 是常数 1
2
Fish C 解释说:
因为
[*]Python 找得到 a,也找得到 b,所以 a+b 不能触发 __radd__方法,故此 Python 按原本的 __add__ 方法运算,得出 8。
[*]反之,
因 Python 在 1+b 中找不到 a,所以就触发了反运算 __radd__方法,得到了结果 2。
提问:
Python 哪里规定过:找不到 a 就触发反运算,反之不触发?费解。我猜想,是不是因为:class 中定义过 self 和 other 两个变量,所以只要运算时出现了两个变量,即不能“触发”?但是,这逻辑,貌似也不通。求教! 本帖最后由 jackz007 于 2021-3-15 23:51 编辑
图片来自 Python 3.8.7 手册
If the right operand’s type is a subclass of the left operand’s type and that subclass provides a different implementation of the reflected method for the operation, this method will be called before the left operand’s non-reflected method. This behavior allows subclasses to override their ancestors’ operations.
百度翻译译文如下:
如果右操作数的类型是左操作数类型的子类,并且该子类为操作提供了反射方法的不同实现,则将在左操作数的非反射方法之前调用此方法。这种行为允许子类覆盖其祖先的操作。
显然,这是 Python 对象的规则策略。
我觉得是因为第一个数不支持加运算,所以使用了第二个数的反运算 >>> class Nint(int):
def __radd__(self, other):
return int.__sub__(self, other)
>>> a = Nint(4)
>>> type(a)
<class '__main__.Nint'>
>>> type(1)
<class 'int'>
>>> 1 + a
3个人理解就是他们类型不一样,不支持加的操作,所以用了第二个数的反运算 柿子饼同学 发表于 2021-3-15 11:00
个人理解就是他们类型不一样,不支持加的操作,所以用了第二个数的反运算
好的,我再等等其他答案哈,谢谢 个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__add__方法,发现1 和 b 不是同一类型,所以无法实现 + 操作,从而使用 b的__radd__ 方法 弈秋呜呜呜 发表于 2021-3-15 18:57
个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__ad ...
感谢指点~ jackz007 发表于 2021-3-15 10:13
图片来自 Python 3.8.7 手册
If the right operand’s type is a subclass of the left o ...
这个英文解释准确,谢谢大佬! 弈秋呜呜呜 发表于 2021-3-15 18:57
个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__ad ...
学习了,我觉得你的这个解释更容易理解些!
页:
[1]