鱼C论坛

 找回密码
 立即注册
查看: 2375|回复: 6

[技术交流] 【Cookbook】保留最后 N 个元素

[复制链接]
发表于 2020-3-9 10:18:55 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
问题
在迭代操作或者其他操作的时候,怎样只保留最后有限几个元素的历史记录?


                               
登录/注册后可看大图


解决方案
保留有限历史记录正是 collections.deque 大显身手的时候。比如,下面的代码在多行上面做简单的文本匹配, 并返回匹配所在行的最后N行:

  1. from collections import deque


  2. def search(lines, pattern, history=5):
  3.     previous_lines = deque(maxlen=history)
  4.     for line in lines:
  5.         if pattern in line:
  6.             yield line, previous_lines
  7.         previous_lines.append(line)

  8. # Example use on a file
  9. if __name__ == '__main__':
  10.     with open(r'../../cookbook/somefile.txt') as f:
  11.         for line, prevlines in search(f, 'python', 5):
  12.             for pline in prevlines:
  13.                 print(pline, end='')
  14.             print(line, end='')
  15.             print('-' * 20)
复制代码


                               
登录/注册后可看大图


讨论
我们在写查询元素的代码时,通常会使用包含 yield 表达式的生成器函数,也就是我们上面示例代码中的那样。 这样可以将搜索过程代码和使用搜索结果代码解耦。

使用 deque(maxlen=N) 构造函数会新建一个固定大小的队列。当新的元素加入并且这个队列已满的时候, 最老的元素会自动被移除掉。

  • 代码示例:

  1. >>> q = deque(maxlen=3)
  2. >>> q.append(1)
  3. >>> q.append(2)
  4. >>> q.append(3)
  5. >>> q
  6. deque([1, 2, 3], maxlen=3)
  7. >>> q.append(4)
  8. >>> q
  9. deque([2, 3, 4], maxlen=3)
  10. >>> q.append(5)
  11. >>> q
  12. deque([3, 4, 5], maxlen=3)
复制代码

尽管你也可以手动在一个列表上实现这一的操作(比如增加、删除等等)。但是这里的队列方案会更加优雅并且运行得更快些。

更一般的, deque 类可以被用在任何你只需要一个简单队列数据结构的场合。 如果你不设置最大队列大小,那么就会得到一个无限大小队列,你可以在队列的两端执行添加和弹出元素的操作。

  • 代码示例:

  1. >>> q = deque()
  2. >>> q.append(1)
  3. >>> q.append(2)
  4. >>> q.append(3)
  5. >>> q
  6. deque([1, 2, 3])
  7. >>> q.appendleft(4)
  8. >>> q
  9. deque([4, 1, 2, 3])
  10. >>> q.pop()
  11. 3
  12. >>> q
  13. deque([4, 1, 2])
  14. >>> q.popleft()
  15. 4
复制代码

在队列两端插入或删除元素时间复杂度都是 O(1) ,区别于列表,在列表的开头插入或删除元素的时间复杂度为 O(N) 。



                               
登录/注册后可看大图

思考题:
怎样从一个集合中获得最大或者最小的 N 个元素列表?

游客,如果您要查看本帖隐藏内容请回复





  1. 摘自《Python Cookbook》第三版
  2. 本翻译项目源地址:<a href="https://github.com/yidao620c/python3-cookbook" target="_blank">https://github.com/yidao620c/python3-cookbook</a>
  3. Copyright (c) 2014-2018 Xiong Neng and other contributors
  4. 遵循Apache License 2.0 协议
复制代码
索引贴

暂无待更新~

由于原项目在线阅读不方便,因此转载给各位鱼油(我才不会说是水经验!)
Cookbook系列帖子正在不断完善哦,希望能对已经初步了解python的你有所帮助~
获取更新欢迎关注淘帖https://fishc.com.cn/forum.php?mod=collection&action=view&ctid=1664


喜欢记得评分哦!

本帖被以下淘专辑推荐:

小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2020-3-10 14:36:02 | 显示全部楼层

回帖奖励 +3 鱼币

感谢分享
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-12 08:15:50 | 显示全部楼层

回帖奖励 +3 鱼币

查看隐藏内容
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-13 13:03:17 | 显示全部楼层

回帖奖励 +3 鱼币

查看答案
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-13 14:41:08 | 显示全部楼层

回帖奖励 +3 鱼币

感谢
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2020-3-13 18:19:54 | 显示全部楼层

回帖奖励 +3 鱼币

我看看
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-15 09:34:13 | 显示全部楼层

回帖奖励 +3 鱼币

还没学到 学习一下
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-6-8 06:07

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表