python中*和**的打包和解包

python中的*和**,能够让函数支持任意数量的参数,它们在函数定义和调用中,有着不同的目的

1|0一. 打包参数


1|1* 的作用:在函数定义中,收集所有的位置参数到一个新的元组,并将这个元组赋值给变量args


>>> def f(*args): print(args) >>> f() () >>> f(1) (1,) >>> f(1, 2, 3, 4) (1, 2, 3, 4) >>>

1|2** 的作用:在函数定义中,收集关键字参数传递给一个字典,并将这个字典赋值给变量kwargs


>>> def f(**kwargs): print(kwargs) >>> f() {} >>> f(a=1, b=2) {'a': 1, 'b': 2} >>>

2|0二. 解包参数


2|1* 的作用:在函数调用中,* 能够将元组或者列表解包成不同的参数


>>> def func(a, b, c, d): print(a, b, c, d) >>> args = (1, 2, 3, 4) >>> func(*args) 1 2 3 4 >>> args = [1, 2, 3, 4] >>> func(*args) 1 2 3 4

2|2** 的作用:在函数调用中,**会以键/值的形式解包一个字典,使其成为独立的关键字参数


>>> def func(a, b, c, d): print(a, b, c, d) >>> kwargs = {"a": 1, "b": 2, "c": 3, "d": 4} >>> func(**kwargs) 1 2 3 4

3|0三. 注意


3|11. 在函数定义时,* 表示打包,在函数体内部, * 仍然表示解包(print(*args)实际上也算是调用了print函数)


>>> def foo(*args, **kwargs): print(args) #未解包参数 print(*args) #解包参数 >>> v = (1, 2, 4) >>> d = {'a':1, 'b':12} >>> foo(v, d) ((1, 2, 4), {'a': 1, 'b': 12}) (1, 2, 4) {'a': 1, 'b': 12}

3|22. 打包和解包并不能脱离函数而存在


表面上看并没有什么函数,实际上是有的,用的就是format的函数调用

 

>>> c = {"name": 'zhang', "age": 2} >>> **c SyntaxError: invalid syntax >>> >>> "Name:{name}, Age:{age}".format(**c) 'Name:zhang, Age:2'

参考源码中对format函数的定义

但是这个字典解包不能用print函数输出

>>> print(**c) Traceback (most recent call last): File "<pyshell#40>", line 1, in <module> print(**c) TypeError: 'age' is an invalid keyword argument for this function >>>

因为上述字典解出来的形式是这样的:

**c = name='zhang',age=2

而print函数只支持*args,不支持**kwargs

3|33. 在ddt中的应用


@ddt.data(*all_caseDatas)中,data是一个函数,调用函数的时候,参数*all_caseDatas自动将参数列表[{}, {}, {}...]解包为{},{},{}...,在def data(*values)函数中,*会自动将各个位置参数打包成新的元组({}, {}, {}...),然后@ddt.data就可以获取每一条数据作为测试用例了

@ddt.data(*all_caseDatas) def test_my_request(self, case_data): global global_var if len(global_var) != 0 and case_data["request_data"] is not None: for key, value in global_var.items(): if case_data["request_data"].find(key) != -1: case_data["request_data"] = case_data["request_data"].replace(key, value)

4|0四. 练习


请写出下列代码的运行结果

def f(str1, *args, **kwargs): print(str1, args, kwargs) l = [1, 2, 3] t = [4, 5, 6] d = {"a":7, "b":8, "c":9} f(1, 2) f(1, 2, 3, "python") f("python", a=1, b=2, c=3) print("================") f("python", l, d) f("python", *t) f("python", *l, **d) f("python", q="winning", **d) 运行结果: 1 (2,) {} 1 (2, 3, 'python') {} python () {'a': 1, 'b': 2, 'c': 3} ================ python ([1, 2, 3], {'a': 7, 'b': 8, 'c': 9}) {} python (4, 5, 6) {} python (1, 2, 3) {'a': 7, 'b': 8, 'c': 9} python () {'a': 7, 'b': 8, 'q': 'winning', 'c': 9}

需要注意的是f("python", *t)是把列表t先解包成 4, 5, 6,然后在def f(str1, *args, **kwargs):中将4, 5, 6重新打包成新的元组(4, 5, 6)再传递给变量args

 


__EOF__

本文作者cnhkzyy
本文链接https://www.cnblogs.com/my_captain/p/9410658.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角推荐一下。您的鼓励是博主的最大动力!
posted @   cnhkzyy  阅读(5360)  评论(0编辑  收藏  举报
编辑推荐:
· 没有源码,如何修改代码逻辑?
· 一个奇形怪状的面试题:Bean中的CHM要不要加volatile?
· [.NET]调用本地 Deepseek 模型
· 一个费力不讨好的项目,让我损失了近一半的绩效!
· .NET Core 托管堆内存泄露/CPU异常的常见思路
阅读排行:
· 微软正式发布.NET 10 Preview 1:开启下一代开发框架新篇章
· C# 集成 DeepSeek 模型实现 AI 私有化(本地部署与 API 调用教程)
· DeepSeek R1 简明指南:架构、训练、本地部署及硬件要求
· 没有源码,如何修改代码逻辑?
· NetPad:一个.NET开源、跨平台的C#编辑器
点击右上角即可分享
微信分享提示