* args和** kwargs是什么意思?

时间:2008-11-13 14:33:52

标签: python

*args**kwargs究竟是什么意思?

根据Python文档,从它看来,它传递了一个参数元组。

def foo(hello, *args):
    print hello

    for each in args:
        print each

if __name__ == '__main__':
    foo("LOVE", ["lol", "lololol"])

打印出来:

LOVE
['lol', 'lololol']

你如何有效地使用它们?

5 个答案:

答案 0 :(得分:245)

*args和/或**kwargs作为函数定义参数列表中的最后一项,允许该函数接受任意数量的参数和/或关键字参数。

例如,如果你想编写一个函数来返回所有参数的总和,无论你提供多少参数,你都可以这样写:

def my_sum(*args):
    return sum(args)

当你覆盖一个函数时,它可能更常用于面向对象的编程,并希望用用户传入的任何参数调用原始函数。

你实际上不必称他们为argskwargs,这只是一个惯例。这是神奇的***

官方Python文档有a more in-depth look

答案 1 :(得分:87)

另外,我们使用它们来管理继承。

class Super( object ):
   def __init__( self, this, that ):
       self.this = this
       self.that = that

class Sub( Super ):
   def __init__( self, myStuff, *args, **kw ):
       super( Sub, self ).__init__( *args, **kw )
       self.myStuff= myStuff

x= Super( 2.7, 3.1 )
y= Sub( "green", 7, 6 )

这样Sub就不知道(或关心)超类初始化是什么。如果您意识到需要更改超类,您可以解决问题,而不必为每个子类中的细节烦恼。

答案 2 :(得分:65)

请注意S.Lott's comment中很酷的事情 - 您还可以使用*mylist**mydict调用函数来解压位置和关键字参数:

def foo(a, b, c, d):
  print a, b, c, d

l = [0, 1]
d = {"d":3, "c":2}

foo(*l, **d)

将打印:0 1 2 3

答案 3 :(得分:25)

*args**kwargs的另一个好用途:你可以定义通用的“catch all”函数,这对于返回这样的包装器而不是原始函数的装饰器来说非常有用。

一个简单的缓存装饰器的例子:

import pickle, functools
def cache(f):
  _cache = {}
  def wrapper(*args, **kwargs):
    key = pickle.dumps((args, kwargs))
    if key not in _cache:
      _cache[key] = f(*args, **kwargs) # call the wrapped function, save in cache
    return _cache[key] # read value from cache
  functools.update_wrapper(wrapper, f) # update wrapper's metadata
  return wrapper

import time
@cache
def foo(n):
  time.sleep(2)
  return n*2

foo(10) # first call with parameter 10, sleeps
foo(10) # returns immediately

答案 4 :(得分:17)

只是为了澄清如何解包参数,并处理缺少的参数等。

def func(**keyword_args):
  #-->keyword_args is a dictionary
  print 'func:'
  print keyword_args
  if keyword_args.has_key('b'): print keyword_args['b']
  if keyword_args.has_key('c'): print keyword_args['c']

def func2(*positional_args):
  #-->positional_args is a tuple
  print 'func2:'
  print positional_args
  if len(positional_args) > 1:
    print positional_args[1]

def func3(*positional_args, **keyword_args):
  #It is an error to switch the order ie. def func3(**keyword_args, *positional_args):
  print 'func3:'
  print positional_args
  print keyword_args

func(a='apple',b='banana')
func(c='candle')
func2('apple','banana')#It is an error to do func2(a='apple',b='banana')
func3('apple','banana',a='apple',b='banana')
func3('apple',b='banana')#It is an error to do func3(b='banana','apple')