你有三个问题:
首先,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 只允许您绑定初始位置参数,它不知道如何绑定第二个而不是第一个。由于您要绑定的参数不是第一个并且无法通过名称访问,因此您没有选择。
不过,对于这种情况,有一个解决方法是编写函数式代码。你只需要operator.methodcaller,而不是functools.partial:
f_func = operator.methodcaller('startswith', '#')
operator.methodcaller 绑定所有参数除了第一个。
既然你问什么是使用functools 的正确方法,我想你可以用一个额外的高阶函数来反转参数:
def flip(func):
return lambda x,y: func(y,x)
f_func = partial(flip(str.startswith), '#')
您可能会认为这是“作弊”,因为它使用了 lambda,如果我们想使用 lambda,我们会像 Daniel 的回答那样使用它。但是我们正在使用它来实现 Python 不提供的基本高阶函数,一旦不妨碍定义f_func 的行感觉很好且实用。
请注意,我并没有在 Python 术语中实现那个高阶函数,尤其是 well,因为我没有像 partial 那样提供任何元数据。它有一个func 只读属性,用户可能也希望flip 也提供该属性。
或者,您可以为 str.startswith 编写一个 Python 包装器,例如上面的 foo,然后将 partial 应用于它。