<p>你不能像类、对象、模块等那样真正地“monkeypatch”一个函数</p>
<p>这些其他事情最终都归结为一个属性集合,因此用不同的属性替换一个属性,或者添加一个新属性,既简单又有用。另一方面,函数基本上是原子的东西。*</p>
<p>当然,您可以通过替换<code>sum</code>函数来monkeypatch内置模块。但我不认为这是你要的。(如果是,请参见下文。)</p>
<p>无论如何,您不能修补<code>sum</code>,但是您可以编写一个新函数,如果需要,可以使用相同的名称(可能在原始函数的周围加上一个包装器,您会注意到,这正是decorator所做的)。在</p>
<hr/>
<p>但实际上没有办法使用<code>sum(['s','t','a','c','k'])</code>来做你想做的事情,因为<code>sum</code>在默认情况下从0开始并向其添加内容。不能向0添加字符串。**</p>
<p>当然,您总是可以传递显式的<code>start</code>,而不是使用默认值,但是您必须更改调用代码以发送适当的<code>start</code>。在某些情况下(例如,您发送的是一个文本列表显示),这是非常明显的;在其他情况下(例如,在一个通用函数中),它可能不是。这在这里仍然不起作用,因为<code>sum(['s','t','a','c','k'], '')</code>只会引发一个<code>TypeError</code>(尝试并阅读错误以了解原因),但它在其他情况下也可以工作。在</p>
<p>但是没有办法避免必须知道一个合适的起始值,因为<code>sum</code>就是这样工作的。在</p>
<p>如果你仔细想想,<code>sum</code>在概念上等同于:</p>
<pre><code>def sum(iterable, start=0):
reduce(operator.add, iterable, start)
</code></pre>
<p>这里唯一真正的问题是<code>start</code>,对吗?<code>reduce</code>允许您不使用start值,它将从iterable中的第一个值开始:</p>
^{pr2}$
<p>这是<code>sum</code>做不到的。但是,如果你真的想,你可以重新定义<code>sum</code>,这样它就可以<em>了:</p>
<pre><code>>>> def sum(iterable):
... return reduce(operator.add, iterable)
</code></pre>
<p>…或:</p>
<pre><code>>>> sentinel = object()
>>> def sum(iterable, start=sentinel):
... if start is sentinel:
... return reduce(operator.add, iterable)
... else:
... return reduce(operator.add, iterable, start)
</code></pre>
<p>但是请注意,这个<code>sum</code>在整数上比原始的慢得多,它将引发一个<code>TypeError</code>,而不是在空序列上返回{<cd18>},依此类推。在</p>
<hr/>
<p>如果您真的想要monkeypatch内置函数(而不是仅仅用一个新名称定义一个新函数,或者在模块的<code>globals()</code>中定义一个新函数来隐藏内置函数),下面是一个适用于python3.1+的示例,只要您的模块使用的是普通的globals字典(除非您在嵌入式解释器或<code>exec</code>调用或类似程序中运行,否则它们将是这样):</p>
<pre><code>import builtins
builtins.sum = _new_sum
</code></pre>
<p>换言之,与修补任何其他模块一样。在</p>
<p>在2.x中,该模块称为<code>__builtin__</code>。在2.3左右和3.0中,关于如何通过全局变量访问的规则发生了变化。有关详细信息,请参见<a href="http://docs.python.org/3/library/builtins.html" rel="nofollow noreferrer">^{<cd22>}</a>/<a href="http://docs.python.org/2/library/__builtin__.html" rel="nofollow noreferrer">^{<cd21>}</a>。在</p>
<hr/>
<p>*当然这不是真的。函数在其代码对象的顶部有一个名称、闭包单元格列表、文档字符串等。甚至代码对象也是一个字节码序列,您可以使用<code>bytecodehacks</code>或硬编码的黑客。除了<code>sum</code>实际上是一个内置函数,而不是一个函数,因此它甚至没有从Python访问的代码……总之,对于大多数目的来说,它已经足够接近了,可以说函数是原子的东西。在</p>
<p>**当然,<em>可以</em>将字符串转换为某些知道如何将自身添加到整数的子类(通过忽略它们),但实际上,您不想这样做。在</p>