Peteryo01223 发表于 2021-3-15 10:13:32

二问: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 10:13:33

本帖最后由 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 对象的规则策略。

柿子饼同学 发表于 2021-3-15 10:57:01

我觉得是因为第一个数不支持加运算,所以使用了第二个数的反运算

柿子饼同学 发表于 2021-3-15 11:00:29

>>> 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个人理解就是他们类型不一样,不支持加的操作,所以用了第二个数的反运算

Peteryo01223 发表于 2021-3-15 13:35:43

柿子饼同学 发表于 2021-3-15 11:00
个人理解就是他们类型不一样,不支持加的操作,所以用了第二个数的反运算

好的,我再等等其他答案哈,谢谢

弈秋呜呜呜 发表于 2021-3-15 18:57:29

个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__add__方法,发现1 和 b 不是同一类型,所以无法实现 + 操作,从而使用 b的__radd__ 方法

Peteryo01223 发表于 2021-3-16 10:02:16

弈秋呜呜呜 发表于 2021-3-15 18:57
个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__ad ...

感谢指点~

Peteryo01223 发表于 2021-3-16 10:03:00

jackz007 发表于 2021-3-15 10:13
图片来自 Python 3.8.7 手册

If the right operand’s type is a subclass of the left o ...

这个英文解释准确,谢谢大佬!

bravsheng 发表于 2021-8-6 17:57:05

弈秋呜呜呜 发表于 2021-3-15 18:57
个人理解 :1 是类型 int ;而 b 是 类型 Nint。不同类型之间是不支持算数运算的。1 + b 中,先调用1 的__ad ...

学习了,我觉得你的这个解释更容易理解些!
页: [1]
查看完整版本: 二问:Py 第43课,算数2,反运算