"在内置数据类型(dict、list、set、tuple)的基础上,collections模块提供了几个额外的数据类型:

  • namedtuple:生成可以使用名字来访问元素内容的tuple,通常用来增强代码的可读性, 在访问一些tuple类型的数据时尤其好用.
  • deque:双端队列,可以快速的从另外一侧追加和推出对象.
  • Counter:计数器,主要用来计数.
  • OrderedDict:有序字典.
  • defaultdict:带有默认值的字典.

@


namedtuple

首先,我们知道tuple可以表示不变集合,例如一个点的二维坐标可以表示为:

>>> p = (1, 2)

可是,看到(1, 2),我们很难分辨出这个tuple是用来表示一个坐标的.

这时,namedtuple就派上用场了:

>>> from collections import namedtuple
>>> Ponint = namedtuple('Ponint', ['x', 'y'])
>>> p = Ponint(1, 2)
>>> p.x
1
>>> p.y
2

再比如,有这样一个数据结构:每一个对象都是拥有三个元素的tuple.

使用namedtuple方法就可以方便的通过tuple来生成可读性更高也更好用的数据结构.

from collections import namedtuple

userinfo = [
("花千骨", "洪荒之力", "白子画"),
("白浅", "玉清昆仑扇", "夜华"),
("锦觅", "翊圣玄冰", "旭凤")
] Userinfo = namedtuple('Userinfo', ["姓名", "必杀技", "男朋友"])
[print(Userinfo._make(ui)) for ui in userinfo] """打印结果如下:
Userinfo(姓名='花千骨', 必杀技='洪荒之力', 男朋友='白子画')
Userinfo(姓名='白浅', 必杀技='玉清昆仑扇', 男朋友='夜华')
Userinfo(姓名='锦觅', 必杀技='翊圣玄冰', 男朋友='旭凤')
"""

deque

所谓的单端队列,就是一端进,另一端出.

所谓的双端队列,就是两端都可以进出.

deque其实是double-ended queue的缩写,翻译过来就是双端队列.

deque最大的好处就是实现了从队列头部快速增加和取出对象:.popleft(), .appendleft()

你可能会说,原生的list也可以从头部添加和取出对象啊?就像这样:

lst.insert(0, 'a')
lst.pop(0)

但是,我们得注意了,list对象的这两种用法的时间复杂度为O(n),也就是说随着元素数量的增加,耗时呈线性上升.

而使用deque对象则是O(1)的复杂度,所以当你的代码有这样的需求的时候,一定要记得使用deque.

deque与list的效率问题

·

  • 使用list存储数据时,按索引访问元素很快,但是插入和删除元素就很慢了,因为list是线性存储,数据量大的时候,插入和删除的效率会很低.
  • 在使用insert、remove方法时,deque的平均效率要远高于list.
  • list根据索引查某个值的效率要高于deque.
  • append和pop方法对于列表的效率没有影响.

示例 deque两端进出

>>> from collections import deque
>>> dq = deque([1, 2, 3])
>>> dq.append('z')
>>> dq.appendleft('a') # 从头部添加元素
>>> dq.pop()
'z'
>>> dq.popleft() # 从头部弹出元素
'a'

作为一个双端队列,deque还提供了其它的好用的方法,比如rotate.

下面是一个有趣的例子,主要使用了deque的rotate方法来实现一个无限循环的加载动画.

from collections import deque
import sys
import time running = deque('>---------------------')
while 1:
print('\r', ''.join(running), end='')
running.rotate(1)
sys.stdout.flush()
time.sleep(0.2) # 一个无限循环的跑马灯
# ------------>--------

Counter

Counter类的目的是用来跟踪值出现的次数.

Counter是一个无序的容器类型,以字典的键值对形式存储,其中元素作为key,计数作为value,计数值可以是任意的Interger(包括0和负数).

Counter类和其它语言的bags或multisets很相似.

示例 使用Counter模块统计一段句子里面所有字符出现次数

from collections import Counter

s = '''
A Counter is a dict subclass for counting hashable objects.
It is an unordered collection where elements are stored as dictionary keys and
their counts are stored as dictionary values.
Counts are allowed to be any integer value including zero or negative counts.
The Counter class is similar to bags or multisets in other languages.
'''.lower() c = Counter(s)
# 获取出现频率最高的4个字符
print(c.most_common(4)) # [(' ', 74), ('e', 32), ('s', 25), ('a', 24)]

OrderedDict

在Python中,dict这个数据结构由于hash的特性,是无序的,这在有的时候会给我们带来一些麻烦.

幸运的是,collections模块为我们提供了OrderedDict,当你要获得一个有序的字典对象时,用它就对了.

Python3.6版本以后的字典是有序的了.

简单示例

>>> from collections import OrderedDict
>>> d = dict([('a', 1), ('b', 2), ('c', 3)])
>>> d # dict的Key是无序的
{'a': 1, 'c': 3, 'b': 2}
>>> od = OrderedDict([('a', 1), ('b', 2), ('c', 3)])
>>> od # OrderedDict的Key是有序的
OrderedDict([('a', 1), ('b', 2), ('c', 3)])

·

但要注意的是,OrderedDict的key是按照插入的顺序排列的,而不是Key本身排序:

>>> od = OrderedDict()
>>> od['z'] = 1
>>> od['y'] = 2
>>> od['x'] = 3
>>> od.keys() # 按照插入的Key的顺序返回
['z', 'y', 'x']

defaultdict

一个示例

·

有这么一个集合:[1, 2, 3, 4, 5, 7, 8, 9]

将所有小于5的值保存至字典的第一个key中,大于5的值保存至第二个key中.

·

我们先来看看dict如何实现:

lst = [1, 2, 3, 4, 5, 7, 8, 9]
dct = {}
for i in lst:
key = 'k1' if i < 5 else 'k2'
dct.setdefault(key, []).append(i)

再来看看defaultdict如何实现:

from collections import defaultdict

lst = [1, 2, 3, 4, 5, 7, 8, 9]
dd = defaultdict(list) for i in lst:
key = 'k1' if i < 5 else 'k2'
dd[key].append(i) # defaultdict(<class 'list'>, {'k1': [1, 2, 3, 4], 'k2': [5, 7, 8, 9]})

还有,在使用Python原生的数据结构dict的时候,如果用d[key]这样的方式访问, 当指定的key不存在时,是会抛出KeyError异常的.

但是,如果使用defaultdict,只要你传入一个默认的工厂方法,那么请求一个不存在的key时, 便会调用这个工厂方法使用其结果来作为这个key的默认值.

>>> from collections import defaultdict
>>> dd = defaultdict(lambda: 'N/A')
>>> dd['name'] = "花千骨"
>>> dd['name']
'花千骨'
>>> dd['sex'] #'sex'不存在,将返回默认值
'N/A'

上面只是非常简单的介绍了一下collections模块的主要内容,主要目的就是当你碰到适合使用它们的场所时,能够记起并使用它们,起到事半功倍的效果.

如果想要对它们有一个更全面和深入了解的话,建议阅读官方文档和模块源码,Go.



"

【Python collections】的更多相关文章

  1. 【Python数据分析】Python3多线程并发网络爬虫-以豆瓣图书Top250为例

    基于上两篇文章的工作 [Python数据分析]Python3操作Excel-以豆瓣图书Top250为例 [Python数据分析]Python3操作Excel(二) 一些问题的解决与优化 已经正确地实现 ...

  2. 【Python数据分析】Python3操作Excel(二) 一些问题的解决与优化

    继上一篇[Python数据分析]Python3操作Excel-以豆瓣图书Top250为例 对豆瓣图书Top250进行爬取以后,鉴于还有一些问题没有解决,所以进行了进一步的交流讨论,这期间得到了一只尼玛 ...

  3. 利用Dnspod api批量更新添加DNS解析【python脚本】 - 推酷

    利用Dnspod api批量更新添加DNS解析[python脚本] - 推酷 undefined

  4. 【python进阶】详解元类及其应用2

    前言 在上一篇文章[python进阶]详解元类及其应用1中,我们提到了关于元类的一些前置知识,介绍了类对象,动态创建类,使用type创建类,这一节我们将继续接着上文来讲~~~ 5.使⽤type创建带有 ...

  5. 【python进阶】Garbage collection垃圾回收2

    前言 在上一篇文章[python进阶]Garbage collection垃圾回收1,我们讲述了Garbage collection(GC垃圾回收),画说Ruby与Python垃圾回收,Python中 ...

  6. 【python进阶】深入理解系统进程2

    前言 在上一篇[python进阶]深入理解系统进程1中,我们讲述了多任务的一些概念,多进程的创建,fork等一些问题,这一节我们继续接着讲述系统进程的一些方法及注意点 multiprocessing ...

  7. 【python图像处理】图像的缩放、旋转与翻转

    [python图像处理]图像的缩放.旋转与翻转 图像的几何变换,如缩放.旋转和翻转等,在图像处理中扮演着重要的角色,python中的Image类分别提供了这些操作的接口函数,下面进行逐一介绍. 1.图 ...

  8. 【Python 开发】Python目录

    目录: [Python开发]第一篇:计算机基础 [Python 开发]第二篇 :Python安装 [Python 开发]第三篇:python 实用小工具

  9. 【Python教程】《零基础入门学习Python》(小甲鱼)

    [Python教程]<零基础入门学习Python>(小甲鱼) 讲解通俗易懂,诙谐. 哈哈哈. https://www.bilibili.com/video/av27789609

随机推荐

  1. Python爬虫连载8-JS加密(一)

    一.JS加密 1.有的反爬虫策略采用js对需要传输的数据进行加密处理. 2.经过加密,传输的就是密文 3.加密函数或者过程一定是在浏览器完成,也就是一定会把代码(js代码)暴露给使用者 4.通多阅读加 ...

  2. Java必须知道的知识点

    junit用法,before,beforeClass,after, afterClass的执行顺序 分布式锁 nginx的请求转发算法,如何配置根据权重转发 用hashmap实现redis有什么问题( ...

  3. 11g RAC添加用户表空间(数据文件)至文件系统(File System)的修正

    前提:非TEMP.UNDO和SYSTEM表空间,这仨是大爷,您得搂着点.来自博客园AskScuti .客户是添加临时表空间数据文件时,不小心 ADD 到了文件系统中,然后发现,后悔了,还在OS层面 R ...

  4. 关于spring boot集成MQTT

    安装 说到mqtt,首先肯定要安装了,安装什么的地址:http://activemq.apache.org/ap...我本地是Windows的环境,所以装的是Windows版本,这里是第一个注意的地方 ...

  5. leetcode全部滑动窗口题目总结C++写法(完结)

    3. 无重复字符的最长子串 A: 要找最长的无重复子串,所以用一个map保存出现过的字符,并且维持一个窗口,用le和ri指针标识.ri为当前要遍历的字符,如果ri字符在map中出现过,那么将le字符从 ...

  6. uGUI源码调试

    uGUI源代码地址:https://bitbucket.org/Unity-Technologies/ui 工具编译后转换位置{Unity3D_Vserion}\Editor\Data\UnityEx ...

  7. .net_DevExpress控件使用经验总结

    (转)DevExpress控件使用经验总结DevExpress是一个比较有名的界面控件套件,提供了一系列的界面控件套件的DotNet界面控件.本文主要介绍我在使用DevExpress控件过程中,遇到或 ...

  8. python面试的100题(4)

    4.打乱一个排好序的list对象alist? import random alist = [1,2,3,4,5] random.shuffle(alist) print(alist) 结果为:[2, ...

  9. STM32F103之ADC学习记录

    1.问题 1)10位ADC的误差是多少? 首先要分清分辨率与精度的区别. 10cm的尺子,有100个等分刻度,则该尺子的分辨率为1mm. 但不能说这把尺子的精度是1mm. 在冬天,尺子会热胀冷缩,依然 ...

  10. python UI自动化生成BeautifulReport测试报告并保存截图

    前面已经写过利用BeautifulReport生成测试报告,那么接下来讲讲如何在测试报告里面保存截图 首先需要在测试用例中定义一个截图的方法: # 截图方法 """ os ...