普通参数与关键字参数

“关键字参数”与常规参数有何不同?所有的参数不能被传递为name=value而不是使用位置语法吗?

285220 次浏览

有两种方法将参数值赋给函数形参,这两种方法都被使用。

  1. 的位置。位置参数没有关键字,首先被赋值。

  2. 通过关键字。关键字参数具有关键字,并且在位置参数之后被赋值。

注意有使用位置参数的选项。

如果不使用位置参数,那么——是的——所写的一切都是一个关键字参数。

调用一个函数时,你决定使用位置或关键字或混合。如果你愿意,你可以选择做所有的关键字。我们中的一些人并没有做出这样的选择,而是使用位置参数。

使用关键字参数与普通参数是一样的,只是顺序不重要。例如,下面两个函数调用是相同的:

def foo(bar, baz):
pass


foo(1, 2)
foo(baz=2, bar=1)

位置参数

它们前面没有关键词。顺序很重要!

func(1,2,3, "foo")

关键字参数

他们在前面有关键词。它们可以是任何顺序!

func(foo="bar", baz=5, hello=123)


func(baz=5, foo="bar", hello=123)

您还应该知道,如果您使用默认参数而忽略了插入关键字,那么顺序将会很重要!

def func(foo=1, baz=2, hello=3): ...
func("bar", 5, 123)

我很惊讶没有人提到你可以混合位置参数和关键字参数,使用*args**kwargs (从这个网站)来做这样的鬼鬼祟祟的事情:

def test_var_kwargs(farg, **kwargs):
print "formal arg:", farg
for key in kwargs:
print "another keyword arg: %s: %s" % (key, kwargs[key])

这允许您使用任意关键字参数,这些参数可能包含您不想在前面定义的键。

还有最后一个重要的语言特性。考虑以下函数:

def foo(*positional, **keywords):
print "Positional:", positional
print "Keywords:", keywords

*positional参数将存储传递给foo()的所有位置参数,不限制您可以提供多少。

>>> foo('one', 'two', 'three')
Positional: ('one', 'two', 'three')
Keywords: {}

**keywords参数将存储任何关键字参数:

>>> foo(a='one', b='two', c='three')
Positional: ()
Keywords: {'a': 'one', 'c': 'three', 'b': 'two'}

当然,你可以同时使用这两个词:

>>> foo('one','two',c='three',d='four')
Positional: ('one', 'two')
Keywords: {'c': 'three', 'd': 'four'}

这些特性很少被使用,但偶尔它们非常有用,知道哪些参数是位置参数或关键字是很重要的。

有两个相关的概念,都称为“关键字参数”。

在调用端,也就是其他评论者提到的,您可以通过名称指定一些函数参数。你必须在所有没有名字的参数之后提到它们(位置参数),并且对于任何根本没有提到的参数必须有默认值

另一个概念是在函数定义方面:您可以定义一个按名称接受参数的函数——您甚至不必指定这些名称是什么。它们是纯关键字参数,并且不能被位置传递。语法是

def my_function(arg1, arg2, **kwargs)

传递给这个函数的任何关键字参数都将被放入一个名为kwargs的字典中。你可以在运行时检查这个字典的键,像这样:

def my_function(**kwargs):
print str(kwargs)


my_function(a=12, b="abc")


{'a': 12, 'b': 'abc'}

我很惊讶,似乎没有人指出可以传递一个键控参数字典,满足形式参数,就像这样。

>>> def func(a='a', b='b', c='c', **kwargs):
...    print 'a:%s, b:%s, c:%s' % (a, b, c)
...
>>> func()
a:a, b:b, c:c
>>> func(**{'a' : 'z', 'b':'q', 'c':'v'})
a:z, b:q, c:v
>>>

使用Python 3,你可以拥有这两个必需的关键字和非必需的关键字参数:


可选:(为参数'b'定义的默认值)

def func1(a, *, b=42):
...
func1(value_for_a) # b is optional and will default to 42

要求(没有为参数'b'定义默认值):

def func2(a, *, b):
...
func2(value_for_a, b=21) # b is set to 21 by the function call
func2(value_for_a) # ERROR: missing 1 required keyword-only argument: 'b'`

这可以帮助在你有许多相似的参数,特别是如果他们是相同类型的情况下,在这种情况下,我更喜欢使用命名参数或我创建一个自定义类,如果参数属于一起。

我正在寻找一个使用类型注释的默认kwargs的示例:

def test_var_kwarg(a: str, b: str='B', c: str='', **kwargs) -> str:
return ' '.join([a, b, c, str(kwargs)])

例子:

>>> print(test_var_kwarg('A', c='okay'))
A B okay {}
>>> d = {'f': 'F', 'g': 'G'}
>>> print(test_var_kwarg('a', c='c', b='b', **d))
a b c {'f': 'F', 'g': 'G'}
>>> print(test_var_kwarg('a', 'b', 'c'))
a b c {}

只需补充/添加一种在调用函数时定义没有在关键字中指定的参数的默认值的方法:

def func(**keywargs):
if 'my_word' not in keywargs:
word = 'default_msg'
else:
word = keywargs['my_word']
return word

叫它:

print(func())
print(func(my_word='love'))

你会得到:

default_msg
love

在python中阅读有关*args**kwargs的更多信息:https://www.digitalocean.com/community/tutorials/how-to-use-args-and-kwargs-in-python-3