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将是一个Barwith val = 24,但是obj2生成错误的赋值因为Bar没有__mul__而且Foo没有__rmul__.
我希望这很清楚.
| 归档时间: |
|
| 查看次数: |
20647 次 |
| 最近记录: |