7

在 Python (2.7) 中,我想创建一个有理数类,它模仿 Fraction 类的行为(在模块分数中)但覆盖 __repr__ 方法以匹配 __str__ 的结果。最初的想法只是为了我自己的修补,只是为了让 IDLE 输出看起来更友好。但现在我更感兴趣的是理解潜在的继承/键入问题——我认为这可能是普遍的兴趣——而不是为这个特定的用例找到解决方法,这无疑是微不足道的。

问题是,我也想继承所有数值运算符(方法 __add__、__sub__ 等)的功能,但结果是我的子类的实例而不是分数的实例。这就是我想要的,但相反,会发生这种情况:

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__ 是首选,但似乎不会捕获我感兴趣的方法调用,因为它们是在基类中定义的!)

鉴于我不是基类的作者,有没有比单独覆盖每个方法更好的方法?

4

1 回答 1

2

这有点工作,但您可以包装它并创建一个委托人。实际上,我做了与您所做的类似的事情,以创建一个默认以十六进制打印的 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
于 2013-11-13T15:35:00.060 回答