Operator commutativity

Ethan Furman ethan at stoneleaf.us
Wed Sep 21 14:15:56 EDT 2011


Mark Dickinson wrote:
> On Sep 21, 2:07 am, Steven D'Aprano <steve
> +comp.lang.pyt... at pearwood.info> wrote:
>> After playing around with various combinations of C1, C2, D1 and D2, it
>> seems to me that the rule is:
>>
>> If the right-hand argument is a subclass of the left-hand argument, AND also
>> defines __radd__ directly rather than inheriting it, then its __radd__
>> method is called before the left-hand argument's __add__ method.
>>
>> which strikes me as a strangely specific and not very useful rule. I suspect
>> it might be an accident of implementation rather than a deliberate feature.
> 
> I'm 99.9% sure it's deliberate rather than an accident of
> implementation.  See the note in the docs at:
> 
> http://docs.python.org/reference/datamodel.html#emulating-numeric-types
> 
> Support that you're subclassing int, (class MyInt(int): ...) and you
> want to intercept additions of the form 3 + MyInt(4)  (perhaps because
> you want MyInt to be 'contagious', so that an arithmetic operation
> that combines an int and a MyInt returns a MyInt).  How would you
> achieve this without this rule?

I think Steven's objection was in not calling subclass.__radd__ when 
__radd__ is inherited rather than directly overridden.  Interestingly 
enough, the following works as expected (at least, as I expected ;)...

class C1(object):
      def __add__(self, other):
          print "C1.__add__(%r, %r)" % (self, other)

class C2(C1):
      def __radd__(self, other):
          print "C2.__radd__(%r, %r)" % (self, other)

class C3(C2):
     pass

C1() + C2()  # --> C2.__radd__(<...C2...>, <...C1...)
C1() + C3()  # --> C2.__radd__(<...C3...>, <...C1...>)
C2() + C3()  # --> C1.__add__(<...C2...>, <...C3...>)


~Ethan~



More information about the Python-list mailing list