python – 是否可以使用__rmod__覆盖str的%行为?

我想这样做:
x %doSomething% y

对于任何x和任何y都很容易做到(见下面的代码),除非x是str.

是否有任何方法(例如添加特殊方法或引发特定错误)导致旧样式字符串格式化失败(类似于1%doSomthing如何通过TypeError失败)并恢复到doSomething对象中定义的__rmod__方法

class BinaryMessage(object):
  def __init__(self,fn):
    self._fn = fn
  def __rmod__(self,LHS):
    return BinaryMessagePartial(self._fn,LHS)

class BinaryMessagePartial(object):
  def __init__(self,fn,LHS):
    self._fn = fn
    self._LHS = LHS
  def __mod__(self,RHS):
    return self._fn(self._LHS,RHS)

def _doSomething(a,b):
  return a + b

doSomething = BinaryMessage(_doSomething)

result = 5 %doSomething% 6
assert result == 11

解决方法

注意:我提交了Python 2.7和3.5及更高版本的补丁.这些已经登陆并且是2.7.14,3.5.4,3.6.1和3.7的一部分,其中OP示例现在按预期工作.对于旧版本,请参阅下文.

不幸的是,这在Python中目前是不可能的.行为在评估循环中是硬编码的:

TARGET(BINARY_MODULO) {
    PyObject *divisor = POP();
    PyObject *dividend = TOP();
    PyObject *res = PyUnicode_CheckExact(dividend) ?
        PyUnicode_Format(dividend,divisor) :
        PyNumber_Remainder(dividend,divisor);

(从Python 3.5 source code开始,PyUnicode是Python str类型).

这是不幸的,因为对于其他所有类型,您可以通过使用右侧操作数的子类来阻止LHS .__ mod__方法;从documentation

Note: If the right operand’s type is a subclass of the left operand’s type and that subclass provides the reflected method for the operation,this method will be called before the left operand’s non-reflected method. This behavior allows subclasses to override their ancestors’ operations.

这将是唯一的选项,str%other永远不会返回NotImplemented,所有RHS类型都被接受(实际的str.__mod__ method只接受RHS的str对象,但在这种情况下不调用).

我认为这是Python中的一个错误,提交为issue #28598.

相关文章

在这篇文章中,我们深入学习了XPath作为一种常见的网络爬虫技巧。XPath是一种用于定位和选择XML文档中特...
祝福大家龙年快乐!愿你们的生活像龙一样充满力量和勇气,愿你们在新的一年里,追逐梦想,勇往直前,不...
今天在爬虫实战中,除了正常爬取网页数据外,我们还添加了一个下载功能,主要任务是爬取小说并将其下载...
完美收官,本文是爬虫实战的最后一章了,所以尽管本文着重呈现爬虫实战,但其中有一大部分内容专注于数...
JSON是一种流行的数据传输格式,Python中有多种处理JSON的方式。官方的json库是最常用的,它提供了简单...
独立样本T检验适用于比较两组独立样本的均值差异,而配对T检验则适用于比较同一组样本在不同条件下的均...