我不理解下面的例子,假设我有这些函数:

# python likes
def save(filename, data, **kwargs):
    fo = openX(filename, "w", **kwargs) # <- #1
    fo.write(data)
    fo.close()
# python doesnt like
def save2(filename, data, **kwargs):
    fo = openX(filename, "w", kwargs) # <- #2
    fo.write(data)
    fo.close()

def openX(filename, mode, **kwargs):
    #doing something fancy and returning a file object

为什么第一条是正确的,第二条是错误的?**kwargs基本上是一个字典,所以如果我想把参数传递给openX,我认为正确的方法是不使用**,只给出字典。但是Python显然不喜欢第二个,它告诉我我给了3个而不是2个参数。

这背后的原因是什么呢?


当前回答

下面的代码使用kwargs并将其转移到另一个函数:

def myprint( kwargs ):
    # default values
    a = kwargs.get('a', None)
    b = kwargs.get('b', None)

    # print both
    print('a={}, b={}'.format(a,b))

def mytest( **kwargs ):
    myprint( kwargs )

mytest()
mytest(b=2)

收益率:

a=None, b=None
a=None, b=2

其他回答

因为字典是一个单独的值。如果希望将其作为一组关键字参数传递,则需要使用关键字展开。

在第二个示例中,您提供了3个参数:文件名、模式和字典(kwargs)。但是Python期望:2个形式参数加上关键字参数。

通过在字典前面加上“**”,可以将字典kwargs解压缩为关键字参数。

字典(类型dict)是包含键值对的单个变量。

“关键字参数”是键值方法参数。

任何字典都可以在函数调用时通过在前缀中加上**来解包到关键字参数。

扩展@gecco的回答,下面是一个例子,将向您展示两者的区别:

def foo(**kwargs):
    for entry in kwargs.items():
        print("Key: {}, value: {}".format(entry[0], entry[1]))

# call using normal keys:
foo(a=1, b=2, c=3)
# call using an unpacked dictionary:
foo(**{"a": 1, "b":2, "c":3})

# call using a dictionary fails because the function will think you are
# giving it a positional argument
foo({"a": 1, "b": 2, "c": 3})
# this yields the same error as any other positional argument
foo(3)
foo("string")

在这里,您可以看到解包字典是如何工作的,以及为什么发送实际的字典会失败

**语法告诉Python将关键字参数收集到字典中。save2将它作为一个非关键字参数(字典对象)传递下去。openX没有看到任何关键字参数,因此**args不会被使用。而是获得第三个非关键字参数(字典)。修改openX函数的定义。

def openX(filename, mode, kwargs):
    pass

下面的代码使用kwargs并将其转移到另一个函数:

def myprint( kwargs ):
    # default values
    a = kwargs.get('a', None)
    b = kwargs.get('b', None)

    # print both
    print('a={}, b={}'.format(a,b))

def mytest( **kwargs ):
    myprint( kwargs )

mytest()
mytest(b=2)

收益率:

a=None, b=None
a=None, b=2