2

我正在使用 functools 模块中的部分方法将函数映射到一系列值:

def basic_rule(p,b,vx=1,**kwargs):
    return (p / b) if vx != 0 else  0

def rule5(func,**kwargs):
    vals = map(functools.partial(func,**kwargs), range(1,kwargs['b']+1))
    return [x for i,x in enumerate(vals[:-1]) if x >= vals[i+1]] == []

rule5(basic_rule,p=100,b=10000)

这是我在第 5 行得到的错误:

----> return map(functools.partial(func,**kwargs), range(1,kwargs['b']+1))

TypeError: basic_rule() got multiple values for keyword argument 'p'

看起来 functools.partial 正在尝试将范围分配给参数 p,即使我已经为它分配了一个值。我正在尝试将范围分配给 vx 的值。知道我怎样才能做到这一点吗?

编辑:在代码中添加了一些额外的上下文。基本上我想要测试 5 做的是确保赋予它的函数的结果随着 vt 的上升而增加,因此 `func(vt=1) < func(vt=2)... < func(vt =n)。

4

1 回答 1

3

functools.partial生成一个将参数receiveids存储在2个属性中的部分:

  • arguments存储位置参数
  • keywords存储所有基于关键字的参数

因此,部分可以完全按照预期调用原始函数。换句话说,当您使用一个参数(比如说 1)调用结果部分时,它将与以下内容相同:

original_func(1, **kwargs)

由于您kwargs包含第一个参数-并且您将“1”作为位置参数传递-您会收到错误消息。

我不确定它是否适用于这种特殊情况,但一种解决方案是使用inspect.getargspec从中提取kwargs可以作为位置参数传递的参数。在这种情况下,rule5函数类似于:

def rule5(func, **kwargs):
    # let's save 'b' argument because we'll need it in the range call
    b = kwargs['b']
    original_args = inspect.getargspec(func).args
    # extract from kwargs all arguments that can be passed as positional
    new_args = [kwargs.pop(key) for key in original_args if key in kwargs]
    # construct the partial passing as much positional arguments as possible
    fn = functools.partial(func, *new_args, **kwargs)
    # now map will pass the range result as the next positional argument
    vals = map(fn, range(1, b+1))
    return [x for i,x in enumerate(vals[:-1]) if x >= vals[i+1]] == []
于 2016-07-03T23:06:19.127 回答