gpt4 book ai didi

python - 使用 str.startswith 的高阶函数

转载 作者:太空宇宙 更新时间:2023-11-04 07:38:31 25 4
gpt4 key购买 nike

我有一个文件,我想从中清除注释行。 我想使用 functools.partial 进行操作,方法类似于以下方式:

from functools import partial

f = open(filetoread, "r")
lines = f.readlines()
f.close()

# Filtering the comment lines
f_func = partial(str.startswith, prefix="#")
lines = filter(f_func, lines)

这不起作用,显然是因为 str.startswith 是一个类方法。将 functools.partialstr.startswith 一起使用的正确方法是什么?

最佳答案

你有三个问题:

首先,str.startswith 不接受关键字参数,它只接受位置参数。它在 C 中实现,并使用 PyArg_ParseTuple 获取其位置参数。没有什么可以告诉 Python 它们有什么名字,因为名字 prefix 只出现在文档字符串中。所以 prefix= 是不行的。问题类似于此:

>>> str.startswith('#hi', prefix='#')
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: startswith() takes no keyword arguments
>>> def foo(self,prefix,start=None,end=None): return str.startswith(self,prefix,start,end)
...
>>> foo('#hi', prefix='#')
True

其次,str.startswith 的第一个参数不是前缀,它是 self,我们在其中检查它是否具有前缀的对象。它作为一种方法并没有内在的问题,如果有一个名为 str.isprefixof 的方法,或者如果这个方法是用 Python 实现的,你会没事的。但没有,现在也没有。

最后,functools.partial 只允许您绑定(bind)初始位置参数,它不知道如何绑定(bind)第二个而不是第一个。由于您要绑定(bind)的参数不是第一个参数并且无法通过名称访问,因此您别无选择。

不过,对于这种情况,有一种变通方法,即编写函数式代码。您只需要 operator.methodcaller,而不是 functools.partial:

f_func = operator.methodcaller('startswith', '#')

operator.methodcaller 绑定(bind)所有参数除了第一个。


既然你问什么是使用functools的正确方法,我想你可以用一个额外的高阶函数来反转参数:

def flip(func):
return lambda x,y: func(y,x)

f_func = partial(flip(str.startswith), '#')

您可能会认为这是“作弊”,因为它使用了 lambda,如果我们想使用 lambda,我们会像 Daniel 的回答那样使用它。但我们正在使用它来实现 Python 不提供的基本高阶函数,一旦它不再妨碍定义 f_func 的行感觉很好且实用。

请注意,我不是在 Python 术语中特别是 实现高阶函数,因为我没有像 partial 那样提供任何元数据.它有一个 func 只读属性,用户可能有理由希望 flip 也提供该属性。

或者,您可以为 str.startswith 编写一个 Python 包装器,如上面的 foo,然后对其应用 partial

关于python - 使用 str.startswith 的高阶函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28695180/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com