元组不可变,你真的确定吗?有了列表,元组存在的意义又是什么?【一文搞懂】

简介: 元组不可变,你真的确定吗?有了列表,元组存在的意义又是什么?【一文搞懂】

一、前言


学过python的小伙伴都知道,python有一种数据类型叫元组,它与列表很相似!

而元组与列表最让我们熟知的区别就在于元组不可变!

类似下面的代码执行的话会直接报错:

a=(1,2)
a[1]=3
print(a)

报错信息bba6e3151dae481386e8dc9bd594c259.png

但元组真的是不可改变的吗?


下面的这段代码可能会让你心生疑惑

我们定义一个元组它的值为{1,{'key':1}},然后改变它的值:

a = (1, {'key': 1})
a[1]['key'] = 2
print(a)

输出结果:

19bb259113ca49a2b35b6955444114ea.png

6d4d3860c49948d6b784186e29e3b88a.jpg


这结果不是改变了吗!!!

那为啥都说元组不可变!!!


二、分析


其实说元组不可变实际上没有错,它指的是元组里元素的内存地址不可改变

我们把改变前和改变后的元素id打印出来,会发现他们是一样的

a = (1, {'key': 1})
print('改变之前的元素id', id(a[1]))
a[1]['key'] = 2
print('改变之后的元素id', id(a[1]))

输出结果

58f9b2a697934057855217bf63564fbc.png


至于为什么修改了元素的值,元素的内存地址未改变。是因为该元素的类型是dict,而dict又是一个可变类型,仅仅改变里面的值的话不会改变其内存地址(详情可参考:python不可变类型和可变类型讲解)


所以在元组中修改它的值是可以的。


三、为什么有了列表还要有不可修改的元组呢?

这里从三个方面做解答

1.元组性能优于列表


元组占用的空间比列表少


我们仅仅定义了6个相同的元素,分配的空间大小就相差了快1倍!

a = [1, 2, 3, 4, 5, 6]
print('列表的空间大小为:', a.__sizeof__())
b = (1, 2, 3, 4, 5, 6)
print('元组的空间大小为:', b.__sizeof__())

输出结果

5c444df35a3d4bed9617fbaec2617a10.png


另外,在Python的垃圾回收机制下,如果一些变量不再使用了,Python会回收它们所占用的内存并返还给操作系统,以便让其他变量或其他应用使用这部分内存。


但是对于一些静态变量,比如元组,当它不被使用并且占用空间不大时,Python 会暂时缓存这部分内存。这样,下次我们再创建同样大小的元组时,Python 就可以不用再向操作系统发出请求,去寻找内存,而是可以直接分配之前缓存的内存空间,这样就能大大加快程序的运行速度!


2.元组可以作为字典key而列表不行


输入下面的代码执行的话会直接报错!


a = [1, 2]
b={a:1}

输出结果

dba3ca0766e040faafd7b51cced3c9ff.png


而如果用元组作为字典的key的话则不会报错!

a = (1, 2)
b={a:1}
print(b)


输出结果


b2fd88242b5545de81dc816f549daf27.png

准确的说,所有不可变类型都可以,而类似list 的可变类型则不可以!!


3.工厂函数


这里就要提到python的一个工厂函数具名元组(namedtuple)


具名元组是 python 标准库 collections 中的工厂函数。它接受两个参数,第一个参数表示类的名称,第二个参数是类的字段名。后者可以是可迭代对象,也可以是空格隔开的字符串。然后,我们通过一串参数的形式将参数传递到构造函数中。这样,我们既可以通过字段名访问元素,也可以用索引访问元素。


只讲具名元组,可能不太好理解。如果称之为带字段名的记录,你可能就清楚了。


现在我们创建一个人员的基础信息类People


它有三个属性:姓名name、性别sex、年龄age


现在创建一个姓名叫张三,22岁的小伙儿,再打印出他的信息

from collections import namedtuple
People = namedtuple('People', 'name sex age')
# 存放在对应字段里的数据要以一串参数的形式传入到构造函数中
person = People('张三', '男', 22)
print(person)
print(person.name)  # 通过字段名获取一个字段的值
print(person._fields)  # 通过fields来获取它有哪些属性


输出结果


3b14348cd2a84cb9862feb024f0a6d19.png


还可以直接将具名元组转化为orderdict(在上述代码末尾加入下列代码)


person_dict = person._asdict()
print(person_dict)


输出结果

06f2bccf1409477b9a6ce426a610cac5.png


四、总结


元组是一种很强大的可以当作记录来用的数据类型,这才是他存在的价值和意义所在!

只是作为一个不可变的列表更被大家所熟知而已。

目录
相关文章
详解Vue3——#default=“scope”
详解Vue3——#default=“scope”
1497 0
|
Arthas Kubernetes 数据可视化
推荐10个GitHub上适合练手的后端项目(涵盖初中高阶)
上周,我们推出了26个好玩又有挑战的前端练习项目。 不少同学留言说,那后端的呢?后端也要! 淘系工程师一呼就应,我们邀请了2位淘系技术后端工程师,筛选出10个难度层层递进,好玩且实用的后端项目,包含java类库中的“瑞士军刀”工具、可视化API展现等等,难度依然分为【初级篇:4个】、【中级篇:3个】、【高级篇:3个】,不同学习诉求的同学可按需选择~
推荐10个GitHub上适合练手的后端项目(涵盖初中高阶)
|
存储 关系型数据库 分布式数据库
【赵渝强老师】HBase的逻辑存储结构
HBase的逻辑存储结构包括命名空间、表和列族。命名空间类似关系型数据库中的数据库,用于逻辑划分和隔离数据;表以RowKey组织数据并按字典序排列,分为多个Region实现分布式存储;列族包含列且无需预先定义,由MemStore缓存写入数据,定期刷新生成Store File。文章通过视频和代码示例详细讲解了各部分的操作与功能。
461 2
|
存储 关系型数据库 分布式数据库
【赵渝强老师】HBase的物理存储结构
本文介绍了HBase的存储结构,包括逻辑与物理存储结构。物理存储主要涉及StoreFile、HFile和HLog日志。HFile是HBase数据存储的核心格式,包含Data块、Meta块、File Info块等六部分,支持压缩以优化存储。HLog(预写日志)记录数据变更,确保数据可靠性,并在Region Server故障时用于恢复。最后,文章详细描述了HBase的写数据流程:先写入WAL日志,再写入MemStore,最终通过Flush操作将数据持久化到HFile中。
707 2
|
JavaScript 前端开发 测试技术
Postman 如何进行性能测试?
Postman 如何进行性能测试?
1871 57
|
Java
把javafx项目打包成exe文件详细过程
本文简化了将JavaFX项目打包成exe文件的过程,首先通过Idea将项目打包成jar包,然后使用GraalVM的native-image工具将jar包编译成exe文件,并展示了执行命令和运行结果。
1183 0
把javafx项目打包成exe文件详细过程
idea+javafx的真正打包方式
本文介绍了使用IntelliJ IDEA进行JavaFX项目打包的正确方法,包括编写一个调用主类的类、引入JavaFX的DLL文件、执行打包操作以及运行打包后的项目的步骤。
1590 0
idea+javafx的真正打包方式
|
Java Apache Maven
BeanUtils库的功能与使用方法详解
BeanUtils库的功能与使用方法详解
|
JavaScript 前端开发 Java
Springboot+vue实现文件的下载和上传
这篇文章介绍了如何在Springboot和Vue中实现文件的上传和下载功能,包括后端控制器的创建、前端Vue组件的实现以及所需的依赖配置。
1806 0
|
Python
pip 安装库失败问题:Retrying (Retry(total=4, connect=None, read=None, redirect=None, status =None)),原因及解决办法
pip 安装库失败问题:Retrying (Retry(total=4, connect=None, read=None, redirect=None, status =None)),原因及解决办法
21071 0