kin*_*all 114
当Python试图将两个对象相乘时,它首先尝试调用左对象的__mul__()
方法.如果左对象没有__mul__()
方法(或者方法返回NotImplemented
,表明它不能与正确的操作数一起使用),那么Python想知道正确的对象是否可以进行乘法.如果右操作数与左边的操作数相同,那么Python知道它不能,因为如果左边的对象不能这样做,那么同一类型的另一个对象当然也不能.
但是,如果这两个对象是不同的类型,Python认为它值得一试.但是,如果操作不是可交换的,它需要某种方式告诉正确的对象它是操作中的正确对象.(乘法当然是,但并非所有运算符都是,并且在任何情况下*
都不总是用于乘法!)所以它调用__rmul__()
而不是__mul__()
.
例如,请考虑以下两个陈述:
print "nom" * 3
print 3 * "nom"
Run Code Online (Sandbox Code Playgroud)
在第一种情况下,Python调用字符串的__mul__()
方法.字符串知道如何将自身乘以整数,所以一切都很好.在第二种情况下,整数不知道如何将自身乘以一个字符串,因此它的__mul()__
返回NotImplemented
和字符串__rmul()__
被调用.它知道该怎么做,你得到的结果与第一种情况相同.
现在我们可以看到,__rmul()__
允许所有字符串的特殊乘法行为包含在str
类中,这样其他类型(如整数)就不需要知道任何有关字符串的信息就可以乘以它们.从现在起一百年后(假设Python仍在使用中),您将能够定义一个新类型,该类型可以按任意顺序乘以整数,即使int
该类在一个多世纪以来一无所知.
顺便说一下,字符串类__mul__()
在某些版本的Python中有一个错误.如果它不知道如何将自己乘以一个对象,它会引发一个TypeError
而不是返回NotImplemented
.这意味着即使用户定义的类型具有__rmul__()
方法,也不能将字符串乘以用户定义的类型,因为字符串永远不会让它有机会.用户定义的类型必须首先(例如,Foo() * 'bar'
而不是'bar' * Foo()
),因此__mul__()
调用它.他们似乎已经在Python 2.7中解决了这个问题(我也在Python 3.2中测试过它),但Python 2.6.6有这个bug.
二元运算符本质上有两个操作数.每个操作数可以位于运算符的左侧或右侧.当您为某种类型的操作符重载时,您可以指定操作符的哪一侧完成重载.在两个不同类型的操作数上调用运算符时,这很有用.这是一个例子:
class Foo(object):
def __init__(self, val):
self.val = val
def __str__(self):
return "Foo [%s]" % self.val
class Bar(object):
def __init__(self, val):
self.val = val
def __rmul__(self, other):
return Bar(self.val * other.val)
def __str__(self):
return "Bar [%s]" % self.val
f = Foo(4)
b = Bar(6)
obj = f * b # Bar [24]
obj2 = b * f # ERROR
Run Code Online (Sandbox Code Playgroud)
这里,obj
将是一个Bar
with val = 24
,但是obj2
生成错误的赋值因为Bar
没有__mul__
而且Foo
没有__rmul__
.
我希望这很清楚.