Python数据结构详解:列表与双端队列
off999 2025-09-01 11:18 43 浏览 0 评论
列表:Python的"万能容器"
在Python的世界里,列表(List)就像是一个功能强大的"万能容器",能够容纳各种类型的数据。它的灵活性和易用性使其成为Python程序员最常用的数据结构之一。
列表的基础操作
创建一个列表非常简单,只需用方括号将元素括起来,元素之间用逗号分隔:
// python
fruits = ['apple', 'banana', 'cherry']
numbers = [1, 2, 3, 4, 5]
mixed = [1, 'hello', 3.14, True]列表支持多种基本操作,包括索引访问、切片、添加元素、删除元素等:
// python
# 索引访问
print(fruits[0]) # 输出: apple
# 切片操作
print(numbers[1:4]) # 输出: [2, 3, 4]
# 添加元素
fruits.append('orange')
print(fruits) # 输出: ['apple', 'banana', 'cherry', 'orange']
# 删除元素
del numbers[2]
print(numbers) # 输出: [1, 2, 4, 5]列表的内部实现
Python列表的内部实现是一个动态数组,这意味着它会根据需要自动调整大小。当列表中的元素数量超过当前容量时,Python会分配一个更大的内存空间,并将所有元素复制到新的空间中。
连排座位示意图
这种实现方式使得列表在随机访问(通过索引访问元素)时非常高效,时间复杂度为O(1)。然而,在列表的开头或中间插入或删除元素时,效率较低,因为需要移动大量元素,时间复杂度为O(n)。
列表的高级用法
列表还支持许多高级操作,如列表推导式、排序、反转等:
// python
# 列表推导式
squares = [x**2 for x in range(10)]
print(squares) # 输出: [0, 1, 4, 9, 16, 25, 36, 49, 64, 81]
# 排序
numbers = [3, 1, 4, 1, 5, 9, 2, 6]
numbers.sort()
print(numbers) # 输出: [1, 1, 2, 3, 4, 5, 6, 9]
# 反转
numbers.reverse()
print(numbers) # 输出: [9, 6, 5, 4, 3, 2, 1, 1]双端队列:高效的双向操作容器
虽然列表在许多情况下都很有用,但在需要频繁在两端进行操作时,双端队列(deque)是一个更好的选择。deque是"double-ended queue"的缩写,它支持在队列的两端高效地添加和删除元素。
deque的基本操作
要使用deque,需要从collections模块中导入:
// python
from collections import deque
# 创建deque
dq = deque(['a', 'b', 'c'])
# 在右端添加元素
dq.append('d')
print(dq) # 输出: deque(['a', 'b', 'c', 'd'])
# 在左端添加元素
dq.appendleft('z')
print(dq) # 输出: deque(['z', 'a', 'b', 'c', 'd'])
# 在右端删除元素
dq.pop()
print(dq) # 输出: deque(['z', 'a', 'b', 'c'])
# 在左端删除元素
dq.popleft()
print(dq) # 输出: deque(['a', 'b', 'c'])deque的内部实现
与列表的动态数组实现不同,deque通常使用双向链表或类似的数据结构实现。这种结构使得在两端添加和删除元素的操作都非常高效,时间复杂度为O(1)。
双向火车示意图
想象一列可以在两端都连接和分离车厢的火车,这就是deque的工作方式。无论你想在前端还是后端添加元素,都可以在常数时间内完成。
deque的高级特性
deque还提供了一些有用的高级特性:
// python
# 限制deque的大小
dq = deque(maxlen=3)
dq.append(1)
dq.append(2)
dq.append(3)
dq.append(4) # 当超过maxlen时,最左边的元素会被自动移除
print(dq) # 输出: deque([2, 3, 4], maxlen=3)
# 旋转操作
dq = deque([1, 2, 3, 4, 5])
dq.rotate(2) # 向右旋转2步
print(dq) # 输出: deque([4, 5, 1, 2, 3])
dq.rotate(-1) # 向左旋转1步
print(dq) # 输出: deque([5, 1, 2, 3, 4])deque的应用场景
deque在许多场景中都非常有用:
- 实现队列和栈 :deque可以同时作为队列(FIFO)和栈(LIFO)使用。
- 滑动窗口 :利用deque的maxlen特性,可以轻松实现滑动窗口算法。
- 缓存 :可以用deque实现一个简单的缓存系统,当缓存满时自动移除最旧的元素。
列表与deque的性能对比
为了更直观地了解列表和deque在不同操作上的性能差异,我们来做一些简单的性能测试。
测试代码
// python
import timeit
from collections import deque
# 测试列表和deque在两端添加元素的性能
def test_append():
list_time = timeit.timeit('lst.append(0)', 'lst = []', number=1000000)
deque_time = timeit.timeit('dq.append(0)', 'from collections import deque; dq = deque()', number=1000000)
print(f"Append to end: List={list_time:.4f}s, Deque={deque_time:.4f}s")
def test_appendleft():
list_time = timeit.timeit('lst.insert(0, 0)', 'lst = []', number=100000) # 列表左端添加操作较慢,减少测试次数
deque_time = timeit.timeit('dq.appendleft(0)', 'from collections import deque; dq = deque()', number=1000000)
print(f"Append to front: List={list_time:.4f}s, Deque={deque_time:.4f}s")
def test_pop():
list_time = timeit.timeit('lst.pop()', 'lst = list(range(1000000))', number=1000000)
deque_time = timeit.timeit('dq.pop()', 'from collections import deque; dq = deque(range(1000000))', number=1000000)
print(f"Pop from end: List={list_time:.4f}s, Deque={deque_time:.4f}s")
def test_popleft():
list_time = timeit.timeit('lst.pop(0)', 'lst = list(range(100000))', number=100000) # 列表左端删除操作较慢,减少测试次数
deque_time = timeit.timeit('dq.popleft()', 'from collections import deque; dq = deque(range(1000000))', number=1000000)
print(f"Pop from front: List={list_time:.4f}s, Deque={deque_time:.4f}s")
test_append()
test_appendleft()
test_pop()
test_popleft()测试结果分析
典型的测试结果可能如下(具体数值会因硬件和软件环境而异):
Append to end: List=0.0523s, Deque=0.0641s
Append to front: List=10.2345s, Deque=0.0782s
Pop from end: List=0.0567s, Deque=0.0612s
Pop from front: List=8.9123s, Deque=0.0654s
从结果中可以看出:
1.在末尾添加和删除元素 :列表和deque的性能相近,列表甚至略快一些。
2.在开头添加和删除元素 :deque的性能远远优于列表,差距可达两个数量级。
这是因为列表在开头操作时需要移动大量元素,而deque可以直接在两端进行操作,不需要移动其他元素。
双端队列的底层数据结构示意图
双端队列示意图
这个示意图展示了双端队列的内部结构,它允许在两端高效地添加和删除元素。每个节点都包含指向前一个和后一个节点的指针,使得整个结构可以像火车一样灵活地在两端扩展或收缩。
如何选择:列表还是deque?
在选择使用列表还是deque时,可以遵循以下原则:
- 随机访问 :如果需要频繁通过索引访问元素,使用列表。
- 两端操作 :如果需要频繁在两端添加或删除元素,使用deque。
- 中间操作 :如果需要频繁在中间插入或删除元素,考虑使用链表结构(Python标准库中没有内置链表,但可以使用其他第三方库)。
- 内存占用 :对于非常大的数据集,deque通常比列表更节省内存。
总结
列表和deque都是Python中强大的序列数据结构,各有其适用场景:
- 列表 :适合需要频繁随机访问元素,或主要在末尾进行添加和删除操作的场景。
- deque :适合需要频繁在两端进行添加和删除操作的场景,如实现队列、栈或滑动窗口等。
通过理解这两种数据结构的内部实现和性能特性,我们可以在实际编程中做出更明智的选择,编写出更高效、更优雅的Python代码。
希望本文能帮助你更好地理解Python中的列表和双端队列,在今后的项目中能够灵活运用它们,提高代码效率和质量。
相关推荐
- 电脑自由截屏的快捷键是什么
-
快捷键是ctrl+alt+a,我们可将聊天窗口缩小,放在旁边。然后找到想要截屏的位置,这时我们在截屏旁边,就更加的方便了。在键盘中按下PrintScreenSysRq(简写为PrtSc)键,此快捷...
- windows10精简版官网下载(win10官方精简版下载)
-
精简版的意思的它比原版的功能和软件少了,其实精简版的更适合大众,没有多余的其他必要功能,更快Win10版本主要为四个分别是专业版、家庭版、企业版、教育版,其实除了这四个之外,还有工作站版、LTSB/L...
- cad2008安装失败(Win11安装cad2008安装失败)
-
解决方法:1、右键点击“开始”按钮,选择“程序和功能”;2、然后点击“启用或关闭windows功能”;3、勾选“Microsoft.NETFramework3.5(包括.Net2.0)”后点击确定按钮...
- u盘在电脑上怎么找出来(u盘在电脑上怎么找到)
-
在电脑中找不到u盘,是因为系统没有自动识别出来,手动打开即可,具体的解决步骤如下:1、在桌面上点击我的电脑,右键,管理。2、打开管理界面,点击储存。3、进到储存页面。4、到这一步,也就可以看到了,有这...
- 联想一体机怎么进入bios(联想一体机怎么进入u盘启动)
-
所需工具:联想Lenovo品牌一体机、启动U盘。具体步骤如下:1、联想一体机从U盘启动设置步骤如下重启联想一体机,启动过程中按F1进入BIOS,部分机型则是开机按Enter键,进入之后再按F12选择进...
- 如何装ghost系统盘(ghost装机教程)
-
ghost是不能做系统c盘,它是一种对硬盘和分区制作成映像文件进行备份和恢复的工具软件,是不能进行操作系统安装。这个软件的使用目的是,当我们安装配置好操作系统以后,用ghost软件对c盘进行备份,或者...
- 加密u盘如何格式化(加密u盘如何格式化手机)
-
1,点击系统与安全进入电脑的控制面板界面,点击上方的系统与安全的选项,在系统界面找到最下方的管理工具功能组。2,选中u盘选择管理工具下面的创建并格式化硬盘分区,点击弹出磁盘管理的界面,在这个里面选中你...
- 万能显卡驱动离线版pc(万能显卡驱动离线版)
-
万用驱动是综合各电脑硬件的性能而制做的软件,对于大多数的电脑硬件驱动都好用,但对于少数品牌电脑驱动要求严格的,就不灵了。有的硬件用万能驱动后,使用效果不佳,就是因为没有完全驱动好。所以,知名品牌电脑硬...
- 笔记本windows8系统下载(笔记本电脑系统win8)
-
在电脑上面就可以下载,打开浏览器搜索windous8系统会出现一些下拉选择,选择第一条或者选择有官网字样的,就直接有下载按钮,然后点击下载就可以了win8可以支持现在可以见到的所有Photosho...
- win 11(win 11 25h2)
-
Windows11是由微软公司(Microsoft)开发的操作系统,应用于计算机和平板电脑等设备。于2021年6月24日发布,2021年10月5日发行。Windows11提供了许多创新...
- 手机视频恢复软件免费版下载
-
手机视频删了怎么恢复 一、安卓手机视频恢复 1.打开电脑,移动鼠标,进入互盾安卓恢复大师官网,下载并安装该软件。手机连接至电脑。手机视频删了怎么恢复 2.打开运行互盾安卓恢复大师,在软件界面看到...
- diy电脑装机教程(diy电脑组装步骤)
-
1,看价格。根据自己的预算价格,选择适合该价格的电脑。注意不要以过高的价格买到配置过低的电脑;2,看性能。根据自己需要的电脑性能,以合理的价格购买。注意不要以过高的价格买到配置过低的电脑。电脑的配置如...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
宝塔面板如何添加免费waf防火墙?(宝塔面板开启https)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
慕ke 前端工程师2024「完整」
-
失业程序员复习python笔记——条件与循环
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
