基本示例.您编写自己的类似int的类:
class FooInt:
... other stuff elided ...
def __sub__(self, other):
if isinstance(other, FooInt):
return self.__class__(self.intvalue - other.intvalue)
elif isinstance(other, int):
return self.__class__(self.intvalue - other)
else:
return NotImplemented
现在,您可以执行以下代码:
FooInt(123) - 456
这很好用; Python在左侧看到FooInt,看到它有__sub__,然后调用FooInt .__ sub __(FooInt(123),456),该返回没有错误,这很好.
接下来我们看到:
123 - FooInt(456)
Python尝试调用int .__ sub __(123,FooInt(456)),但int不知道如何处理FooInt,并返回NotImplemented;它不知道intvalue具有可用于此目的的值.此时,Python不能只调用FooInt .__ sub __(FooInt(456),123),因为它不能假定减法是可交换的(实际上,在大多数数值系统中,在这种情况下,减法不是可交换的)不能只交换运算符的左右两侧并获得正确的结果).这就是__rsub__存在的原因;它使您可以在告诉其他类两件事的同时检查另一个类以寻求一种处理操作的方法:
>位于二进制运算符的右侧(允许它正确处理可交换性)
>左侧的项目不知道如何处理该操作,因此这是正确解决该问题的最后机会
第二点也很重要.在实现__xxx __(左手运算符)时,您希望对识别的类型非常保守.如果您不知道事实是使用已知的正确处理程序来处理已知的具体类型,则不应尝试处理未知类型;另一种类型可能知道如何正确执行该操作,因此您返回NotImplemented并让另一端进行处理.当您实现__rxxx__时,您是最后一次机会.另一个人不知道该怎么做,因此,如果您有任何处理方法,则您应该放任自流,并尽力而为.您可以在Implementing the arithmetic operations的Python文档中看到这一点; __xxx__操作检查具体类型(对于小数,它检查小数,整数,浮点数和复数),而__rxxx__运算符检查常规接口(数字,整数,数字,有理数,数字,实数等).