首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >问答首页 >如何增强python内置函数的特性?

如何增强python内置函数的特性?
EN

Stack Overflow用户
提问于 2013-06-14 06:42:26
回答 5查看 258关注 0票数 1

我是python编程的新手,我想知道如何增强内置函数(Monkeypatch)的特性。

例如

我知道sum()内置函数只允许在数字项上使用

代码语言:javascript
运行
复制
>>> sum([4,5,6,7]) #22

我想让sum函数允许将项目列表作为字符串,如下所示

例如

代码语言:javascript
运行
复制
>>> sum(['s','t','a','c','k']) # 'stack'

提前感谢

EN

回答 5

Stack Overflow用户

回答已采纳

发布于 2013-06-14 07:02:14

不完全是猴子修补,只是重新定义了sum,使其也适用于字符串。

代码语言:javascript
运行
复制
>>> import __builtin__
def sum(seq, start = 0):
    if all(isinstance(x,str) for x in seq):
        return "".join(seq)
    else:
        return __builtin__.sum(seq, start)
...     
>>> sum([4,5,6,7])
22
>>> sum(['s','t','a','c','k'])
'stack'
票数 5
EN

Stack Overflow用户

发布于 2013-06-14 06:59:35

你不能像对类、对象、模块等进行"monkeypatch“。

这些其他的事情最终都归结为一组属性,所以用不同的属性替换一个属性,或者添加一个新的属性,既简单又有用。另一方面,函数基本上是原子的东西。

当然,您可以通过替换sum函数来修补内置模块。但我不认为这是你要问的。(如果您是,请参见下面的内容。)

无论如何,您不能修补sum,但是您可以编写一个新函数,如果您愿意,可以使用相同的名称,(可能在原始函数周围使用包装器-您将注意到,这正是装饰者所做的)。

但是,实际上没有办法使用sum(['s','t','a','c','k'])来做您想做的事情,因为sum默认从0开始并向其添加内容。并且不能将字符串添加到0。**

当然,您可以始终传递显式的start,而不是使用默认值,但您必须更改调用代码以发送适当的start。在某些情况下(例如,在发送文字列表显示的情况下),这很明显;在其他情况下(例如,在泛型函数中),可能不是这样。这在这里仍然不起作用,因为sum(['s','t','a','c','k'], '')只会引发一个TypeError (尝试一下并阅读错误以了解原因),但它在其他情况下可以起作用。

但是,无法避免必须知道sum的适当起始值,因为这就是sum的工作方式。

如果你仔细想想,sum在概念上等同于:

代码语言:javascript
运行
复制
def sum(iterable, start=0):
    reduce(operator.add, iterable, start)

这里唯一真正的问题是start,对吧?reduce允许你去掉起始值,它将从迭代中的第一个值开始:

代码语言:javascript
运行
复制
>>> reduce(operator.add, ['s', 't', 'a', 'c', 'k'])
'stack'

这是sum做不到的。但是,如果您真的想这样做,您可以重新定义sum,这样它就可以

代码语言:javascript
运行
复制
>>> def sum(iterable):
...     return reduce(operator.add, iterable)

…或者:

代码语言:javascript
运行
复制
>>> sentinel = object()
>>> def sum(iterable, start=sentinel):
...     if start is sentinel:
...         return reduce(operator.add, iterable)
...     else:
...         return reduce(operator.add, iterable, start)

但请注意,在处理整数时,这个sum将比原来的慢得多,并且它将引发一个TypeError,而不是在空序列上返回0,依此类推。

如果你真的想修改内置代码(而不是仅仅定义一个新函数,或者在你的模块globals()中定义一个同名的新函数来遮蔽内置代码),下面是一个适用于Python 3.1+的例子,只要你的模块使用普通的全局字典(除非你在一个嵌入式解释器或者exec调用中运行):

代码语言:javascript
运行
复制
import builtins
builtins.sum = _new_sum

换句话说,与monkeypatching任何其他模块一样。

在2.x中,该模块称为__builtin__。关于如何通过全局变量访问它的规则在2.3左右发生了变化,在3.0中又发生了变化。详情请参见builtins/__builtin__

*当然,这并不完全正确。函数在其代码对象之上有名称、闭包单元格列表、文档字符串等。甚至代码对象也是一个字节码序列,您可以对其使用bytecodehacks或硬编码技术。除了sum实际上是一个内置函数,而不是一个函数,所以它甚至没有可从Python…访问的代码无论如何,对于大多数目的来说,它足够接近了,可以说函数是原子的东西。

**当然,您可以将字符串转换为某个知道如何将其自身添加到整数中的子类(通过忽略它们),但实际上,您并不想这样做。

票数 12
EN

Stack Overflow用户

发布于 2013-06-14 06:44:15

要做你想做的事情,你应该使用str.join

代码语言:javascript
运行
复制
"".join(['s','t','a','c','k'])

在Python中,猴子补丁是可能的,但不受欢迎,特别是对于像这样的琐碎事情。这会让你的代码更难读,因为标准库函数会做一些意想不到的事情。

但是,如果你真的想这样做,你可以重新定义这个函数。Python不会阻止你:

代码语言:javascript
运行
复制
def sum(l):
    return "".join(l)

Python将允许您对现有模块做任何您想做的事情:

代码语言:javascript
运行
复制
import sys
sys.stdout = open("somefile", "w")

但再说一次,你不应该这样做。

票数 3
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/17098423

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档