你不知道的Python容器
你不知道的Python容器
昨天阅读了《Python Tricks: The Book》的第五章“Common Data Structures in Python”,该章节介绍了字典、数组、集合、栈、队列以及堆等数据结构的用法和注意事项,其中ChainMap、MappingProxyType等不常使用的容器类引起了我的注意。本文主要对几种不常使用的容器类进行介绍,通过示例说明这些容器的特点。
散列表
ChainMap
“collections.ChainMap”能有效整合多个字典的信息,提供一个可进行查询和更新的视图。它使用列表管理字典的引用,而非创建一个新的字典,因此时空开销比较小。当查询字典内容时,ChainMap按序遍历字典列表,直至找到给定的键;当更新(包括新增、删除)字典内容时,ChainMap只在字典列表的首项上进行操作。
由于ChainMap是从前到后依次查找字典列表,如果有多个字典包含了相同的键,那么靠近列表起始位置的字典就会被优先匹配,而靠近列表末端的字典会被忽略。ChainMap能够为键查找操作设置优先级的特性使得它能够用于管理应用程序的配置项。具体来说,当应用程序具有多种来源(命令行、配置文件、环境变量等)的配置时,可以把优先级高的配置当作ChainMap的第一个位置参数:
from collections import ChainMap
cli_config = { # 命令行参数
'batch_size': 128, # 批次大小为128
'learning_rate': 1e-3,
'dropout': 0.2,
}
default_config = { # 默认配置
'batch_size': 64, # 这里也配置了批次大小
'num_layers': 2,
}
lookup = ChainMap(cli_config, default_config)
lookup['batch_size'] # 128,命令行参数优于匹配
lookup['num_layers'] # 2,默认配置也能查询
lookup.maps # ChainMap({'batch_size': 128, 'learning_rate': 0.001, 'dropout': 0.2}, {'batch_size': 64, 'num_layers': 2})
lookup['beam_size'] = 4 # 新增键值对
del lookup['batch_size'] # 删除键值对
cli_config # {'learning_rate': 0.001, 'dropout': 0.2, 'beam_size': 4},cli_config是字典列表的首项,更新操作会作用到它身上
MappingProxyType
“types.MappingProxyType”通过代理模式控制了对字典内容的访问,它接收一个字典作为参数,返回该字典的只读(Read Only)视图,任何尝试更新该视图的操作都会触发“TypeError”异常。
from types import MappingProxyType
meetings = {
'ACL': 'Annual Meeting of the Association for Computational Linguistics',
'CVPR': 'IEEE Conference on Computer Vision and Pattern Recognition',
}
mapping = MappingProxyType(meetings)
mapping['ACL'] # 'Annual Meeting of...'
del mapping['ACL'] # TypeError: 'mappingproxy' object does not support item deletion
mapping['ACL'] = '' # TypeError: 'mappingproxy' object does not support item assignment
线性表
Python的内置类型“list”支持按位置插入、删除列表元素,基于它可以实现栈和队列等操作受限的线性表,但考虑到“list”本身是基于动态数组的,频繁增删列表元素会使得“list”需要时常调整占用的存储空间(扩容、缩容),最终导致程序性能的下降。
“collections.deque”是实现栈和队列的更好选择,它基于双向链表,支持快速地从序列两端添加或删除元素。
使用deque模拟栈“后进先出”的特性
from collections import deque stack = deque()
stack.append('F')
stack.append('E') stack.pop() # 'E',首先弹出'E'
stack.pop() # 'F',出栈顺序与入栈顺序相反
stack.pop() # IndexError: pop from an empty deque
使用deque模拟队列“先进先出”的特性
from collections import deque queue = deque()
queue.append('F')
queue.append('E') queue.popleft() # 'F','F'首先出队
queue.popleft() # 'E',出队顺序与入队顺序相同
queue.popleft() # IndexError: pop from an empty deque
堆
二叉堆是一棵完全二叉树,有最小堆和最大堆两种类型,其中最小堆的每个节点都小于等于它的子节点。Python的“heapq”模块提供了关于构造最小堆、插入删除元素并保持最小堆特性的接口。
import random
import heapq
digits = [random.randrange(100) for i in range(5)] # [75, 28, 93, 79, 57]
heapq.nsmallest(2, digits) # [28, 57],获取列表中最小的2个元素
heapq.heapify(digits) # 堆化,[28, 57, 93, 79, 75]
heapq.heappush(digits, 10) # 插入一个元素,[10, 57, 28, 79, 75, 93]
heapq.heappop(digits) # 10,弹出最小的元素
“queue.Priority”给出了优先队列的一种实现,它基于最小堆:
from queue import PriorityQueue
q.put((3, 'Sing'))
q.put((1, 'Jump'))
q.put((2, 'Rap'))
while not q.empty():
next_item = q.get()
print(next_item)
# (1, 'Jump'), (2, 'Rap'), (3, 'Sing')
参考资料
- Python Tricks: The Book
- Python's ChainMap: Manage Multiple Contexts Effectively
- 《流畅的Python》,2.9.4节“双向队列和其他形式的队列”
你不知道的Python容器的更多相关文章
- python容器数据类型的特色
python容器数据类型的特色 list: 可变数据类型(不可哈希), 有序, 可索引获取, 可修改 Dict: 可变数据类型(不可哈希), 3.6版本有序, 可通 ...
- Python 学习笔记(1)Python容器:列表、元组、字典与集合
Python容器:列表.元组.字典与集合 列表: 1.列表 的创建 使用[ ] 或者 list()创建列表:empty_list = [ ] 或者 empty_list= list() 使用list( ...
- python容器类型:列表,字典,集合等
容器的概念我是从C++的STL中学到的 什么是容器? 容器是用来存储和组织其他对象的对象. 也就是说容器里面可以放很多东西,这些东西可以是字符串,可以是整数,可以是自定义类型,然后把这些东西有组织的存 ...
- 第三章 Python容器:列表、元组、字典与集合
数据结构的分类依据?基本的"数组"在python中是列表, 数据结构的作用?容器,盛放数据,是由原子组成的分子.可以将一群数据进行整合.拆分.重排. 3.2 列表 列表是啥?顺 ...
- Docker运行python容器
容器是镜像运行的实例,而镜像保存在仓库里,测试或者发布生产环境只需要pull下来即可,相对传统的应用部署,能很好的保持环境的一致,节省运维时间.最近公司内部的java和.net服务也已经全部容器化,实 ...
- python容器类型的协议
1.协议(Protocols)与其他编程语言中的接口恒相似,它规定你哪些地方必须要定义,然而在python中的协议就显得不那么正式,事实上,在python中,协议更像是一种指南. 2.序列类型和映射类 ...
- python 容器 生成器 迭代器 总结
一.容器 容器是一种把多个元素组织在一起的数据结构,容器中的元素可以逐个地迭代获取,可以用in, not in关键字判断元素是否包含在容器中.通常这类数据结构把所有的元素存储在内存中. >> ...
- Python 容器使用的 5 个技巧和 2 个误区
"容器"这两个字很少被 Python 技术文章提起.一看到"容器",大家想到的多是那头蓝色小鲸鱼:Docker,但这篇文章和它没有任何关系.本文里的容器,是 P ...
- Python 容器用法整理
本文整理几种基本容器:列表.元组.字典和集合的用法和collections中几种已经预先实现的容器数据结构:namedtuple(),双向链表deque,ChainMap,Counter,Ordere ...
随机推荐
- JavaScript 事件循环(1) —— 从 setTimeout 说起
转变认知 setTimeout 可能是很多前端工程师爱用的方法,它可以使得一段代码延迟执行,例如: setTimeout(() => console.log('A'), 1000); // 在1 ...
- Java有没有goto?
goto是Java中的保留字,暂时还不是Java的关键字.
- switch语句能否作用在byte上,能否作用在long上,能否作用在String上?
在Java 7以前,在switch(expr1)中,expr1只能是一个整数表达式(但不包括long和Long)或者枚举常量,整数表达式可以是int基本类型或Integer包装类型,byte.shor ...
- maven常用命令含义
今天在开发过程中,对一个mapper.xml文件的sql进行了改动,重启tomcat后发现没有生效,首先考虑是不是远程服务开启着,导致代码没有走本地,确认远程服务是关闭的,的确是本地修改没有生效,于是 ...
- PACT 在微服务架构中的用途是什么?
PACT 是一个开源工具,允许测试服务提供者和消费者之间的交互,与合同隔离, 从而提高微服务集成的可靠性. 微服务中的用法 用于在微服务中实现消费者驱动的合同. 测试微服务的消费者和提供者之间的消费者 ...
- ubuntu 20.04 安装 ros1 和ros2
ubuntu 选择Hong Kong 源 1. ROS1安装 添加 sources.list(设置你的电脑可以从 packages.ros.org 接收软件.) sudo sh -c '. /etc ...
- Python中的numpy库介绍!
转自:https://blog.csdn.net/codedz/article/details/82869370 机器学习算法中大部分都是调用Numpy库来完成基础数值计算的.安装方法: pip3 i ...
- Wepy-小程序踩坑记
引言 用过原生开发的小程序也知道除了api 其他功能性的内容并不多对于需要做大型项目来说是比较难入手的,因此朋友推荐的wepy我就入坑鸟...这么一个跟vue的开发方式类似的框架,不过说起来跟vue类 ...
- 从路由原理出发,深入阅读理解react-router 4.0的源码
react-router等前端路由的原理大致相同,可以实现无刷新的条件下切换显示不同的页面.路由的本质就是页面的URL发生改变时,页面的显示结果可以根据URL的变化而变化,但是页面不会刷新.通过前 ...
- DRF 视图组件
目录 DRF 视图组件 视图组件大纲 两个视图基本类 五个扩展类 九个子类视图 视图集 常用视图集父类 魔法类 一览表 DRF中视图的"七十二变" 第一层:基于APIview的五个 ...