可以将可变数量的参数传递给函数吗?

Posted

技术标签:

【中文标题】可以将可变数量的参数传递给函数吗?【英文标题】:Can a variable number of arguments be passed to a function? 【发布时间】:2009-05-28 07:50:10 【问题描述】:

类似于在 C 或 C++ 中使用可变参数:

fn(a, b)
fn(a, b, c, d, ...)

【问题讨论】:

我将尊敬的 Lotness 推荐给播客 53:itc.conversationsnetwork.org/shows/… 我得和洛特先生一起做这个。您可以在 Python 文档中快速获得有关此问题的权威答案,此外,您还会对文档中的其他内容有所了解。如果您打算使用 Python 工作,那么了解这些文档对您有利。 @DavidSykes 链接已损坏 @DaveLiu ***.blog/2013/10/28/podcast-53-lets-go-rio 【参考方案1】:

是的。您可以将*args 用作非关键字 参数。然后,您将能够传递任意数量的参数。

def manyArgs(*arg):
  print "I was called with", len(arg), "arguments:", arg

>>> manyArgs(1)
I was called with 1 arguments: (1,)
>>> manyArgs(1, 2, 3)
I was called with 3 arguments: (1, 2, 3)

如您所见,Python 会将参数解包为包含所有参数的单个元组。

对于关键字参数,您需要接受它们作为单独的实际参数,如Skurmedel's answer 所示。

【讨论】:

同样重要...人们可能会发现有时他们必须将未知数量的参数传递给函数。在这种情况下,通过创建一个名为“args”的列表并将其传递给 manyArgs 来调用您的“manyArgs”,例如“manyArgs(*args)” 这很接近,但不幸的是这还不够通用:manyArgs(x = 3) 失败并显示TypeError。 Skumedel 的回答显示了解决方案。关键是函数的一般签名是f(*list_args, **keyword_args)(不是f(*list_args))。 args 的类型是tuplekwargs 表示关键字参数kwargs 的类型是dictionary 只需for arg in args: 用于遍历传递的参数 我必须传递可变数量的整数参数,后跟可选字符串,我可以调用 use xxx(*arg: int, log: str = "") 吗?参数数量的变化很小,0、1、2或3,所以我可以使用重载但它似乎在python中不可用。【参考方案2】:

添加到放松帖子:

您也可以发送多个键值参数。

def myfunc(**kwargs):
    # kwargs is a dictionary.
    for k,v in kwargs.iteritems():
         print "%s = %s" % (k, v)

myfunc(abc=123, efh=456)
# abc = 123
# efh = 456

您可以将两者混合使用:

def myfunc2(*args, **kwargs):
   for a in args:
       print a
   for k,v in kwargs.iteritems():
       print "%s = %s" % (k, v)

myfunc2(1, 2, 3, banan=123)
# 1
# 2
# 3
# banan = 123

它们必须按顺序声明和调用,即函数签名需要是 *args、**kwargs,并按此顺序调用。

【讨论】:

不确定你是如何让 myfunc(abc=123, def=456) 工作的,但在我的 (2.7) 中,如果没有得到 SyntaxError,则无法在此函数中传递 'def'。我认为这是因为 def 在 python 中有意义。尝试 myfunc(abc=123,fgh=567) 代替。 (否则,很好的答案,谢谢!) @Dannid:也不知道哈哈...在 2.6 或 3.2 上也不起作用。我会重命名它。 当你说'按那个顺序被叫'时,你的意思是按那个顺序通过吗?还是别的什么? @NikhilPrabhu:是的。调用时,关键字参数必须放在最后。如果您在调用中也有非关键字参数,它们总是排在最后。 对于 Python 3:只需将 print aprint(a) 交换,并将 kwargs.iteritems():kwargs.items() 交换。【参考方案3】:

如果可以的话,Skurmedel 的代码适用于 python 2;要使其适应 python 3,请将 iteritems 更改为 items 并将括号添加到 print。这可以防止像我这样的初学者碰到: AttributeError: 'dict' object has no attribute 'iteritems' 并在其他地方搜索(例如 Error “ 'dict' object has no attribute 'iteritems' ” when trying to use NetworkX's write_shp())为什么会发生这种情况。

def myfunc(**kwargs):
for k,v in kwargs.items():
   print("%s = %s" % (k, v))

myfunc(abc=123, efh=456)
# abc = 123
# efh = 456

和:

def myfunc2(*args, **kwargs):
   for a in args:
       print(a)
   for k,v in kwargs.items():
       print("%s = %s" % (k, v))

myfunc2(1, 2, 3, banan=123)
# 1
# 2
# 3
# banan = 123

【讨论】:

【参考方案4】:

添加到其他优秀的帖子。

有时你不想指定参数的数量并且想为它们使用键(如果在方法中没有使用字典中传递的一个参数,编译器会报错)。

def manyArgs1(args):
  print args.a, args.b #note args.c is not used here

def manyArgs2(args):
  print args.c #note args.b and .c are not used here

class Args: pass

args = Args()
args.a = 1
args.b = 2
args.c = 3

manyArgs1(args) #outputs 1 2
manyArgs2(args) #outputs 3

然后你可以做类似的事情

myfuns = [manyArgs1, manyArgs2]
for fun in myfuns:
  fun(args)

【讨论】:

【参考方案5】:
def f(dic):
    if 'a' in dic:
        print dic['a'],
        pass
    else: print 'None',

    if 'b' in dic:
        print dic['b'],
        pass
    else: print 'None',

    if 'c' in dic:
        print dic['c'],
        pass
    else: print 'None',
    print
    pass
f()
f('a':20,
   'c':30)
f('a':20,
   'c':30,
   'b':'red')
____________

上面的代码会输出

None None None
20 None 30
20 red 30

这就像通过字典传递变量参数一样好

【讨论】:

这是糟糕的代码。更好的是:f = lambda **dic: ' '.join(dic.get(key, 'None') for key in 'abc')【参考方案6】:

除了已经提到的好答案之外,另一种解决方法取决于您可以按位置传递可选的命名参数这一事实。例如,

def f(x,y=None):
    print(x)
    if y is not None:
        print(y)

产量

In [11]: f(1,2)
1
2

In [12]: f(1)
1

【讨论】:

以上是关于可以将可变数量的参数传递给函数吗?的主要内容,如果未能解决你的问题,请参考以下文章

如何将带有 args 的成员函数作为参数传递给另一个成员函数?

怎么将可变参数的函数的参数传递给另一个可变参数的函数

C – 将可变数量的参数传递给 main

如何将传递给我的 bash 脚本的所有参数传递给我的函数? [复制]

我可以将数组作为参数传递给 Java 中具有可变参数的方法吗?

将可变参数函数模板参数传递给另一个函数