Python的函数定义中可以在参数里添加**kwargs——简单来说目的是允许 添加不定参数名称的参数 ,并作为 字典 传递参数。但前提是——你必须提供 参数名 。 例如下述情况: 1 class C()
Python的函数定义中可以在参数里添加**kwargs——简单来说目的是允许添加不定参数名称的参数,并作为字典传递参数。但前提是——你必须提供参数名。
例如下述情况:
1 class C(): 2 def __init__(self, **kwargs): 3 print(kwargs)
有如下输入:
In [48]: c = C() {} In [49]: c = C(a = 1) {‘a‘: 1}
这一切都符合常理。但是当我使用一个字典传递的时候:
In [50]: c = C({‘a‘: 1}) --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-50-2be6d5be2a84> in <module>() ----> 1 c = C({‘a‘: 1})
我一开始以为是__init__太娇气了,换成一般办法:
1 In [51]: class C(): 2 ...: def f(self, **kwargs): 3 ...: print(kwargs) 4 ...:
In [52]: c = C() In [53]: c.f({‘a‘: 1}) --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-53-5daee03baab1> in <module>() ----> 1 c.f({‘a‘: 1}) TypeError: f() takes 1 positional argument but 2 were given
包括定义在__main__下的函数也是如此:
In [54]: def f1(**kwargs): ...: print(kwargs) ...: In [56]: f1({‘a‘: 1}) --------------------------------------------------------------------------- TypeError Traceback (most recent call last) <ipython-input-56-8652a6e75162> in <module>() ----> 1 f1({‘a‘: 1}) TypeError: f1() takes 0 positional arguments but 1 was given
但是当我在f1参数前加一个参数名:
1 In [57]: f1(b = {‘a‘: 1}) 2 {‘b‘: {‘a‘: 1}}
问题立刻解决。
思考:
我们在先前的错误尝试中,一直传入的是个单纯的字典。虽然**kwargs提供将参数转成字典的功能,但是直接传入字典并不会使其理解成参数。实质上Python解释器认为我们输入的是作为参数值的字典,而并未携带参数名称,而Python函数机制中“固定名称”的参数应写在**kwargs的前面,因此误以为我们传入了一个“固定名称”参数,事实上我们先前的定义中并未加入“固定名称”参数,所以报错takes 1 positional arguments but 2 was given。
如果真的要传入字典怎么办呢?这就简单了,直接把**kwargs改成kwargs即可:
1 In [58]: class C(): 2 ...: def __init__(self, kwargs): 3 ...: print(kwargs) 4 ...: 5 6 In [59]: c = C({‘a‘: 1}) 7 {‘a‘: 1}
这次经历使我加深了对**kwargs特殊参数的理解。以后的学习更要注意细节!