在Python(2.7)中,我想创建一个有理数类,以模仿Fraction类的行为(在模块分数中),但重写__repr__方法以匹配__str__的结果。最初的想法只是出于我自己的想法,只是为了让IDLE输出看起来更友好。但是现在,我对了解潜在的继承/类型问题(我认为可能是普遍感兴趣的)更感兴趣,而不是针对此特定用例找到解决方法,这是很简单的。
问题是,我也想继承所有数值运算符的功能(方法__add __,__ sub__等),但是结果是我的子类的实例,而不是Fraction的实例。那就是我想要的,但是发生了:
class Q(Fraction):
def __repr__(self):
return str(self)
>>> Q(1,2)
1/2
>>> Q(1,2) + Q(1,3)
Fraction(5, 6)
发生这种情况是因为Fraction中定义的运算符返回Fraction实例。当然,我可以单独覆盖所有这些魔术方法,调用父类进行数学运算然后强制转换为我的类型,但是我觉得应该有一种方法可以通用地处理这种重复情况(即,无需编写“ def” 20次)。
我还考虑过使用__getattribute__来拦截方法调用,但这似乎很不雅致,极其脆弱,并且可以保证在甚至比此复杂得多的情况下失败。(我知道__getattr__是首选,但似乎不会捕获我感兴趣的方法调用,因为它们是在基类中定义的!)
鉴于我不是基类的作者,是否有比单独覆盖每个方法更好的方法?
这有点工作,但是您可以包装它并创建一个委托人。实际上,我做了与您所做的类似的事情,以创建一个默认情况下以十六进制打印的int。我自己的一个子类中的一个更好的例子,该子类将int子类化,以允许按位读取(写操作显然是行不通的,因为int是不可变的,因此此特定代码并没有走得太远……)。也许有很多示例代码,但是显示了如何使用它:
# I stole this decorator from another stackoverflow recipe :)
def returnthisclassfrom(specials):
specialnames = ['__%s__' % s for s in specials.split()]
def wrapit(cls, method):
return lambda *a: cls(method(*a))
def dowrap(cls):
for n in specialnames:
method = getattr(cls, n)
setattr(cls, n, wrapit(cls, method))
return cls
return dowrap
def int_getslice(self, i, j):
# NON-pythonic, will return everything inclusive i.e. x[5:3] returns 3 bits, not 2.
# Because that's what users normally expect.
# If you're a purist, modify below.
if i > 1000000 or j > 1000000:
raise ValueError, 'BitSize limited to 1 million bits'
lo = min(i,j)
hi = max(i,j)
mask = (1<<((hi-lo)+1))-1
return (self>>lo) & mask
def int_getitem(self, i):
# Safety limit
if i > 1000000:
raise ValueError, 'BitSize limited to 1 million bits'
return (self>>i)&1
def int_iter(self):
# since getitem makes it iterable, override
raise AttributeError, 'int object is not iterable'
@returnthisclassfrom('abs add and div floordiv lshift mod mul neg or pow radd rand rdiv rdivmod rfloordiv rlshift rmod rmul ror rpow rrshift rshift rsub rxor rtruediv sub truediv xor trunc')
class BitSliceInt(long):
__getslice__ = int_getslice
__getitem__ = int_getitem
__iter__ = int_iter
本文收集自互联网,转载请注明来源。
如有侵权,请联系[email protected] 删除。
我来说两句