Python中可变参数 *args 和 **kwargs的用法


【前言】关于python中所有参数的知识:python的位置参数、默认参数、关键字参数、可变参数区别。本文只讲其中的两种可变参数。

【注意】

  • 所有python函数的返回值若有大于一个,则该返回值就是一个tuple:
return (batch_pred,batch_true,correct_num,bound_correct_num)
return batch_pred,batch_true,correct_num,bound_correct_num # 与上面完全等价,事实上,python就是按照tuple处理的

打印函数的类型,output:<class 'tuple'>

  • 首先语法是和**,但是用args和**kwargs只是规定,为了使用方便,但是并没有强制使用它们.

* 允许传递任意数量的参数

当你不确定你的函数里将要传递多少参数时你可以用*args。例如,它可以传递任意数量的参数:

>>> def print_everything(*args):
        for count, thing in enumerate(args):
...         print '{0}. {1}'.format(count, thing)
...
>>> print_everything('apple', 'banana', 'cabbage')
0. apple
1. banana
2. cabbage

** 允许传递没有事先定义的参数名

>>> def table_things(**kwargs):
...     for name, value in kwargs.items():
...         print '{0} = {1}'.format(name, value)
...
>>> table_things(apple = 'fruit', cabbage = 'vegetable')
cabbage = vegetable
apple = fruit

命名参数、解析顺序

你也可以混着用。命名参数首先获得参数值,然后所有的其他参数都传递给args和kwargs。命名参数在列表的最前端。例如:
def table_things(titlestring, **kwargs)
*args和
kwargs可以同时在函数的定义中,但是
args必须在**kwargs前面。

* 还可以解包列表(或者元组)并把它们的每一项传递到下一层参数中

>>> def print_three_things(a, b, c):
...     print 'a = {0}, b = {1}, c = {2}'.format(a,b,c)
...
>>> mylist = ['aardvark', 'baboon', 'cat']
>>> print_three_things(*mylist)

a = aardvark, b = baboon, c = cat

正如*负责将tuple解析一样,**将dict解析。传入一个关键字参数的函数中,将dict1={k:2,v:4}这样的mapping一项项解析开,传入函数中。

重要:*调用时传入方式和解析方式

def f1d(x, *args):
    x0=x #先按位置参数接受第一个参数[8,8,9];后面的一律进入*args的接收范围 统统打包进一个元组里

    print(args) #t就是一个元组。
    
    a=f2(args)    #args传入进来就是元组,因此,这样传给f2,对应的p一定接收到的是元组;
    print('\n')
    
    f3(args)      #如果传入f3,它继续作为*可变参数传入的,因此它将再次被()包裹一次,成为新的元组((9,),)
    print('\n')
    
    print('*args表示解析操作:',*args)  # *表示解析该可变参数,即解析该元组;因此以单个元素的方式打印出来的
    print('\n')
    
    f2(*args)     # 解析后传入f2,相当于分别传入了9,10; 其中,9对应上了p,10作为可变参数对用上了*args,被打包传入f2。


def f2(p,*args):
    print('p in f2',p)
    print('args in f2',args)
    

def f3(*args):
    print('args in f3',args)
    print('args解析后将少一层() in f3',*args)
#     print('妄图再解析一次但不行 in f3',**t) #因为args传入的时候就是(9, 10)了,所以最原始的输入类型就是元组了。不能再使用可变参数的解析方法再解析了

f1d([8,8,9],9,10)

output:
(9, 10) # 打印元组t

p in f2 (9, 10)#p接受args
args in f2 () #此时p接收了元组args:(9, 10); 因此f2里的*args无接收

args in f3 ((9, 10),) # 因为f3只有args参数,所以本来就是元组的9, 10)作为f3的args传入的,所以再次被打包,变成了((9, 10),)

*args表示解析操作: 9 10

p in f2 9
args in f2 (10,)

参考

posted @ 2021-11-15 16:58  zae  阅读(525)  评论(0编辑  收藏  举报