理解 python 中的* 打包和解包

devtools/2024/10/19 23:32:30/

注意, 本文的打包和解包的概念有点类似与java 的装箱和解箱

但是还是有点区别
java 中的装箱指的是 把一系列的值让放入1个对象的属性中

python的打包概念 包括

  1. 把一组元素打包成1个list or tuple
  2. 把一系列 key value pair 打包成1个dict




Python * 的基本功能, 乘法

* 首先是1个人乘法符号 例如

python">from loguru import loggerdef process():logger.info(3 * 12) # 36logger.info("hello" * 3) # hellohellohellologger.info([1, 2, 3] * 3) # [1, 2, 3, 1, 2, 3, 1, 2, 3]logger.info(["a", "b", "c"] * 3) # ['a', 'b', 'c', 'a', 'b', 'c', 'a', 'b', 'c']logger.info("hello" * "world") # TypeError: can't multiply sequence by non-int of type 'str'if __name__ == "__main__":import src.configs.configprocess()

直接的注意的是, * 的乘法不单止 数字乘数字
还包括字符串 乘 整数, 数组乘整数
但是不support string * string , string * float , string * 数组, 数组 * 数组等




* 的打包功能

第1个例子
python">def sample1():list_num = [1, 2, 3]first, second, rest = list_numlogger.info("first is {}".format(first)) # 1logger.info("second is {}".format(second)) # 2    logger.info("rest is {}".format(rest)) # 3

上面的例子就是一个简单的解包例子
我们用3个变量去接受1个3个元素的数组是可以正常给3个变量赋值的

2024-05-05 23:53:25.407 | INFO     | __main__:sample1:6 - first is 1
2024-05-05 23:53:25.407 | INFO     | __main__:sample1:7 - second is 2
2024-05-05 23:53:25.407 | INFO     | __main__:sample1:8 - rest is 3
第2个例子

这时如果我改写成让3个变量去接受6个元素的数组
是会报错的

python">def sample2():list_num = [1, 2, 3, 4, 5, 6, 7]first, second, rest = list_numlogger.info("first is {}".format(first)) # 1logger.info("second is {}".format(second)) # 2    logger.info("rest is {}".format(rest)) # too many values to unpack (expected 3)

错误

  File "/home/gateman/Projects/python/python_common_import/src/pack_unpack/star_pack.py", line 5, in sample1first, second, rest = list_num^^^^^^^^^^^^^^^^^^^
ValueError: too many values to unpack (expected 3)
第3个例子

这时我们可以用*rest 去接受剩下的元素, 这时的rest 就是1个数组, 完成了打包的功能

python">def sample3():list_num = [1, 2, 3, 4, 5, 6, 7]first, second, *rest = list_numlogger.info("first is {}".format(first)) # 1logger.info("second is {}".format(second)) # 2    logger.info("rest is {}".format(rest)) # [3, 4, 5, 6, 7]

输出

2024-05-06 00:01:58.114 | INFO     | __main__:sample3:13 - first is 1
2024-05-06 00:01:58.114 | INFO     | __main__:sample3:14 - second is 2
2024-05-06 00:01:58.114 | INFO     | __main__:sample3:15 - rest is [3, 4, 5, 6, 7]

还有, 尝试打包两个变量是不允许的

python">first, second, *rest, *rest2 = list_num # Only one unpack operation allowed in listPylance  
第4个例子

*打包功能最常用的场景 *args (类似于java 的可变参数 String… strings)

python">def sample4():print_list([1, 2, 3])print_list2(1, 2, 3)def print_list(list_num):for i in list_num:logger.info(i)def print_list2(*list_num):for i in list_num:logger.info(i)if __name__ == "__main__":import src.configs.configsample4()

输出

2024-05-06 00:18:04.939 | INFO     | __main__:print_list:31 - 1
2024-05-06 00:18:04.939 | INFO     | __main__:print_list:31 - 2
2024-05-06 00:18:04.939 | INFO     | __main__:print_list:31 - 3
2024-05-06 00:18:04.939 | INFO     | __main__:print_list2:35 - 1
2024-05-06 00:18:04.939 | INFO     | __main__:print_list2:35 - 2
2024-05-06 00:18:04.940 | INFO     | __main__:print_list2:35 - 3

上面定义了两个输出数组的函数
其中print_list1 接受的1个是普通的数组对象

而print_list2 接受的是多个元素, 而且是可以动态打包到 args 变量的
这就是我们常见到的
def func(*args) *args的意思, 自动把传入的动态多个元素参数 打包称1个数组参数




** 的打包功能 - 打包key value pair 到字典

python">def sample5():print_value("abc", 2, 3, a=4,b=5)def print_value(*args, **kwargs):for i in range(len(args)):logger.info("args[{}] is {}".format(i, args[i]))for key in kwargs:logger.info("kwargs[{}] is {}".format(key, kwargs[key]))if __name__ == "__main__":import src.configs.configsample5()

这个例子中, print_value的参数 是 *args, **kwargs

意思是除了用*args 接受所有的值类型参数(注意上面的例子是string 和 int 参数混合传入的), 还可以用**kwargs来接受剩余的 key value pair 参数, 方便!

输出:

python">2024-05-06 00:31:33.215 | INFO     | __main__:print_value:44 - args[0] is abc
2024-05-06 00:31:33.216 | INFO     | __main__:print_value:44 - args[1] is 2
2024-05-06 00:31:33.216 | INFO     | __main__:print_value:44 - args[2] is 3
2024-05-06 00:31:33.216 | INFO     | __main__:print_value:46 - kwargs[a] is 4
2024-05-06 00:31:33.216 | INFO     | __main__:print_value:46 - kwargs[b] is 5





C
现在讲到解包了

第一个例子:
python">def sample1():person_list = ["Jay", 2]# print_value(person_list) # error print_value() missing 1 required positional argument: 'age'print_value(*person_list)person_dict = {"name": "Jade", "age": 3}# person_dict = {"firstname": "Jade", "age": 3} # error print_value() got an unexpected keyword argument 'firstname'print_value(**person_dict)def print_value(name, age):logger.info("name is {}".format(name))logger.info("age is {}".format(age))if __name__ == "__main__":import src.configs.configsample1()

上面定义了1个普通 print_value() 里面有两个参数
如果直接把数组 person_list 传入肯定是报错的
但是这时可以用 person_list 解包成多个 元素, 所以用person_list 传入是ok的

同样地, **person_dict , 双星可以为dict 对象解包, 但是对象的属性必须等于参数定义的参数名字!

第2个例子

解包同样常见于list/tuple/dict 合并

python">def sample2():list = [1, 2, 3]tuple = (4, 5, 6)tuple2 = (*list, *tuple)logger.info(tuple2) # (1, 2, 3, 4, 5, 6)dict1 = {"name": "Jay", "age": 2}dict2 = {"weight": 20, "height": 85}dict3 = {**dict1, **dict2}logger.info(dict3) # {'name': 'Jay', 'age': 2, 'weight': 20, 'height': 85}if __name__ == "__main__":import src.configs.configsample2()

是不是很方便?
注意的是 list 和 tuple 可以互相合并

但是list 和 tuple 都不能和dict 合并




Python 打包和 解包的常用使用情景

打包:
1. 获取从动态多个参数称list/tuple or dict

解包:
2. 把list/tuple or dict 传入 接受普通值参数的函数
3. 合并容器


http://www.ppmy.cn/devtools/34251.html

相关文章

Django中如何让页面之间建立关系

今天给大家讲解两种让页面建立联系的方式 一、重定向 二、表单提交 先看第一种方式,重定向 首先需要了解客户端发起请求的过程 1、客户端向服务端发起请求,比如请求地址是:http://127.0.0.1:8000/lili/submit/ 2、程序根据路由找到视图函数 3、执行视…

力扣每日一题112:路径总和

题目 简单 给你二叉树的根节点 root 和一个表示目标和的整数 targetSum 。判断该树中是否存在 根节点到叶子节点 的路径,这条路径上所有节点值相加等于目标和 targetSum 。如果存在,返回 true ;否则,返回 false 。 叶子节点 是…

QPS(Queries Per Second)和TPS(Transactions Per Second)的介绍和区别

QPS(Queries Per Second)和TPS(Transactions Per Second)是衡量计算系统性能的两个指标,它们分别代表了系统每秒可以处理的查询数和事务数。虽然这两个术语在某些情况下可以互换使用,但它们在技术上有所区别…

【软件开发规范篇】JAVA后端开发编程规范

作者介绍:本人笔名姑苏老陈,从事JAVA开发工作十多年了,带过大学刚毕业的实习生,也带过技术团队。最近有个朋友的表弟,马上要大学毕业了,想从事JAVA开发工作,但不知道从何处入手。于是&#xff0…

关于YOLO8学习(四)模型转换为ncnn

前文 关于YOLO8学习(一)环境搭建,官方检测模型部署到手机 关于YOLO8学习(二)数据集收集,处理 关于YOLO8学习(三)训练自定义的数据集 简介 本文将会讲解: (1)如何通过PyCharm,进行pt模型的转换,最后输出一个适合手机端使用的模型 开发环境 win10、python 3.11…

【机器学习】BK- SDM与LCM的融合策略在文本到图像生成中的应用

突破边缘设备限制:BK-SDM与LCM的融合策略在文本到图像生成中的应用 一、引言二、稳定扩散算法的挑战与现状三、BK-SDM与LCM的融合策略利用高质量图像-文本对进行训练为LCM量身定制高级蒸馏过程 四、结论与展望 一、引言 随着人工智能技术的飞速发展,文本…

安卓中常见的UI控件

TextView(文本视图)EditText(编辑文本)Button(按钮)ImageView(图像视图)ImageButton(图像按钮)CheckBox(复选框)RadioButton&#xff…

java里的i/o流

在Java中,I/O(输入/输出)流是用于处理输入和输出操作的抽象概念。Java的I/O库提供了许多类和方法,用于从各种来源(如文件、网络、内存等)读取数据(输入流),以及将数据写入…