【问题标题】:Python - list of function/argument tuplesPython - 函数/参数元组列表
【发布时间】:2012-08-06 16:08:55
【问题描述】:
def f1(n): #accepts one argument
    pass

def f2(): #accepts no arguments
    pass

FUNCTION_LIST = [(f1,(2)), #each list entry is a tuple containing a function object and a tuple of arguments
                 (f1,(6)),
                 (f2,())]

for f, arg in FUNCTION_LIST:
    f(arg)

在循环的第三次循环中,它尝试将一个空的参数元组传递给一个不接受任何参数的函数。它给出了错误TypeError: f2() takes no arguments (1 given)。前两个函数调用正常工作 - 元组的 content 被传递,而不是元组本身。

去掉有问题的列表条目中的空参数元组并不能解决问题:

FUNCTION_LIST[2] = (f2,)
for f,arg in FUNCTION_LIST:
    f(arg)

结果为@​​987654324@。

我也尝试过迭代索引而不是列表元素。

for n in range(len(FUNCTION_LIST)):
    FUNCTION_LIST[n][0](FUNCTION_LIST[n][1])

这在第一种情况下给出相同的TypeError,当列表的第三个条目是(f2,)时给出IndexError: tuple index out of range

最后,星号符号也不起作用。这次它在调用f1时出错:

for f,args in FUNCTION_LIST:
    f(*args)

TypeError: f1() argument after * must be a sequence, not int

我已经没有东西可以尝试了。我仍然认为第一个应该工作。谁能指出我正确的方向?

【问题讨论】:

    标签: python list function arguments tuples


    【解决方案1】:

    作为记录,我后来发现的一个不错的替代方法是使用functools.partial。下面的代码做了我想做的事情:

    from functools import partial
    
    def f1(n): #accepts one argument
        pass
    
    def f2(): #accepts no arguments
        pass
    
    FUNCTION_LIST = [partial(f1,2), #each list entry is a callable with the argument pre-ordained
                     partial(f1,6),
                     partial(f2)] #the call to partial is not really necessary for the entry with no arguments.
    
    for f in FUNCTION_LIST: f()
    

    【讨论】:

      【解决方案2】:

      问题是这样的符号:

      (6)
      

      计算为整数值,你需要元组,所以这样写:

      (6, )
      

      你的星号符号会成功。

      【讨论】:

        【解决方案3】:

        您在此代码 sn-p 中的注释显示了与此上下文相关的误解:

        FUNCTION_LIST = [(f1,(2)), #each list entry is a tuple containing a function object and a tuple of arguments
                         (f1,(6)),
                         (f2,())]
        

        表达式(2)(6) 不是元组——它们是整数。您应该使用(2,)(6,) 来表示您想要的单元素元组。修复此问题后,您的循环代码应如下所示:

        for f, args in FUNCTION_LIST:
            f(*args)
        

        请参阅 Python 教程中的 Unpacking Argument Lists,了解对 *args 语法的解释。

        【讨论】:

        • 感谢您的出色回答。推论:是否可以对函数的返回值执行操作?假设我想将第一个条目加倍:FUNCTION_LIST = [(f1*2,(2,)), (f1,(6,)),(f2,())] 行不通,不是吗,因为您试图将乘法运算符应用于函数 object,而不是函数返回的值.
        • 相关:是否可以在参数元组中调用其他函数,这些函数将在调用函数时进行评估? FUNCTION_LIST = [(f1,(time.time(),)), (f1,(6,)),(f2,())] 不会给我一些时间相关的东西,因为对 time.time() 的调用将在填充列表时进行评估。
        • @poorsod:对于延迟执行,你应该定义函数来做你想做的事情。例如def double_f1(x): return 2 * f1(x)def f1_time(): return f1(time.time)。在某些情况下,您可以使用 lambda 函数,但我建议谨慎使用。
        【解决方案4】:

        尝试传递*() 而不是()* 符号告诉 python 解包它后面的迭代,因此它解包空元组并且不向函数传递任何内容,因为元组是空的。

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 2012-04-16
          • 2019-05-20
          • 2021-12-06
          • 2015-01-05
          • 2020-04-27
          • 1970-01-01
          • 1970-01-01
          • 2019-02-23
          相关资源
          最近更新 更多