[Python进阶] 1. 列表与元组

[Python进阶] 1. 列表与元组

一、基础知识
  1. 列表和元组,都是一个可以放置任意数据类型的有序集合,列表或元组中的元素数据类型不需要保持一致。
  2. 列表是mutable而元组是immutable
  3. Python中的列表和元组都支持负数索引,-1表示最后一个元素。
  4. 列表和元组都支持切片操作,且都可以随意嵌套。
  5. 两者可以通过 list()tuple() 函数相互转换。
列表和元组常用的内置函数

count(item) 表示统计列表 / 元组中 item 出现的次数。
index(item) 表示返回列表 / 元组中 item 第一次出现的索引。
list.reverse()list.sort() 分别表示原地倒转列表和排序(注意,元组没有内置的这两个函数)。
reversed()sorted() 同样表示对列表 / 元组进行倒转和排序,但是会返回一个倒转后或者排好序的新的列表 / 元组。

二、列表和元组的存储方式
>>> l = [1, 2, 3]
>>> l.__sizeof__()
64
>>> tup = (1, 2, 3)
>>> tup.__sizeof__()
48

上面结果表明对于相同元素的元组与列表,元组的存储空间比列表要少 16 字节。

  • 由于列表是动态的,所以它需要存储指针,来指向对应的元素,int型指针占8个字节。
  • 由于列表可变,所以需要额外存储已经分配的长度大小(8 字节),这样才可以实时追踪列表空间的使用情况,当空间不足时,及时分配额外空间。
>>> l = []
>>> l.__sizeof__() #空列表的存储空间为 40 字节
40
>>> l.append(1)
>>> l.__sizeof__()
72 #加入了元素 1 之后,列表为其分配了可以存储 4 个元素的空间 (72 - 40)/8 = 4
>>> l.append(2)
>>> l.__sizeof__()
72 #由于之前分配了空间,所以加入元素 2,列表空间不变
>>> l.append(3)
>>> l.__sizeof__()
72
>>> l.append(4)
>>> l.__sizeof__()
72
>>> l.append(5)
>>> l.__sizeof__()
104 #加入元素5之后,列表的空间不足,所以又额外分配了可以存储 4 个元素的空间

列表空间分配:为了减小每次增加 / 删减操作时空间分配的开销,Python 每次分配空间时都会额外多分配一些,这样的机制 (over-allocating)保证了其操作的高效性:增加 / 删除的时间复杂度均为 O(1)。

三、列表和元组的性能

由存储方式的区别可以得出结论:元组要比列表更加轻量级一些,所以总体上来说,元组的性能速度要略优于列表。

Python 会在后台,对静态数据做一些资源缓存(resource caching)。通常来说, 因为垃圾回收机制的存在,如果一些变量不被使用了,Python 就会回收它们所占用的内存,返还给操作系统,以便其他变量或其他应用使用。
但是对于一些静态变量,比如元组,如果它不被使用并且占用空间不大时,Python 会暂时缓存这部分内存。这样,下次我们再创建同样大小的元组时,Python 就可以不用再向操作系统发出请求,去寻找内存,而是可以直接分配之前缓存的内存空间,这样就能大大加快程序的运行速度。

  1. 计算初始化一个相同元素的列表和元组分别所需的时间。可以看到,元组的初始化速度,要比列表快 5 倍。

    python3 -m timeit 'x=(1,2,3,4,5,6)'
    20000000 loops, best of 5: 13.5 nsec per loop
    python3 -m timeit 'x=[1,2,3,4,5,6]'
    5000000 loops, best of 5: 67.7 nsec per loop
    
  2. 如果是索引操作的话,两者的速度差别非常小,几乎可以忽略不计。

    python3 -m timeit -s 'x=[1,2,3,4,5,6]' 'y=x[3]'
    10000000 loops, best of 5: 30.1 nsec per loop
    python3 -m timeit -s 'x=(1,2,3,4,5,6)' 'y=x[3]'
    10000000 loops, best of 5: 30.2 nsec per loop
    
  3. 如果增加、删减或者改变元素,那么显然列表更优,因为元组要重新开辟内存空间。

四、列表和元组的使用场景
  1. 元组适用于存储的数据和数量不变。
  2. 列表适用于存储的数据或数量是可变的。
五、总结

列表是动态的,长度可变,可以随意的增加、删减或改变元素。列表的存储空间略大于元组,性能略逊于元组。
元组是静态的,长度大小固定,不可以对元素进行增加、删减或者改变操作。元组相对于列表更加轻量级,性能稍优。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值