Python实现一个优先级队列的方法


Posted in Python onJuly 31, 2020

问题

怎样实现一个按优先级排序的队列? 并且在这个队列上面每次 pop 操作总是返回优先级最高的那个元素

解决方案

下面的类利用 heapq 模块实现了一个简单的优先级队列:

import heapq

class PriorityQueue:
 def __init__(self):
  self._queue = []
  self._index = 0

 def push(self, item, priority):
  heapq.heappush(self._queue, (-priority, self._index, item))
  self._index += 1

 def pop(self):
  return heapq.heappop(self._queue)[-1]

下面是它的使用方式:

>>> class Item:
...  def __init__(self, name):
...   self.name = name
...  def __repr__(self):
...   return 'Item({!r})'.format(self.name)
...
>>> q = PriorityQueue()
>>> q.push(Item('foo'), 1)
>>> q.push(Item('bar'), 5)
>>> q.push(Item('spam'), 4)
>>> q.push(Item('grok'), 1)
>>> q.pop()
Item('bar')
>>> q.pop()
Item('spam')
>>> q.pop()
Item('foo')
>>> q.pop()
Item('grok')
>>>

仔细观察可以发现,第一个 pop() 操作返回优先级最高的元素。 另外注意到如果两个有着相同优先级的元素( foogrok ),pop 操作按照它们被插入到队列的顺序返回的。

讨论

这一小节我们主要关注 heapq 模块的使用。 函数 heapq.heappush() heapq.heappop() 分别在队列 _queue 上插入和删除第一个元素, 并且队列 _queue 保证第一个元素拥有最高优先级( 1.4 节已经讨论过这个问题)。 heappop() 函数总是返回”最小的”的元素,这就是保证队列pop操作返回正确元素的关键。 另外,由于 push 和 pop 操作时间复杂度为 O(log N),其中 N 是堆的大小,因此就算是 N 很大的时候它们运行速度也依旧很快。

在上面代码中,队列包含了一个 (-priority, index, item) 的元组。 优先级为负数的目的是使得元素按照优先级从高到低排序。 这个跟普通的按优先级从低到高排序的堆排序恰巧相反。

index 变量的作用是保证同等优先级元素的正确排序。 通过保存一个不断增加的 index 下标变量,可以确保元素按照它们插入的顺序排序。 而且, index 变量也在相同优先级元素比较的时候起到重要作用。

为了阐明这些,先假定 Item 实例是不支持排序的:

>>> a = Item('foo')
>>> b = Item('bar')
>>> a < b
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: unorderable types: Item() < Item()
>>>

如果你使用元组 (priority, item) ,只要两个元素的优先级不同就能比较。 但是如果两个元素优先级一样的话,那么比较操作就会跟之前一样出错:

>>> a = (1, Item('foo'))
>>> b = (5, Item('bar'))
>>> a < b
True
>>> c = (1, Item('grok'))
>>> a < c
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: unorderable types: Item() < Item()
>>>

通过引入另外的 index 变量组成三元组 (priority, index, item) ,就能很好的避免上面的错误, 因为不可能有两个元素有相同的 index 值。Python 在做元组比较时候,如果前面的比较已经可以确定结果了, 后面的比较操作就不会发生了:

>>> a = (1, 0, Item('foo'))
>>> b = (5, 1, Item('bar'))
>>> c = (1, 2, Item('grok'))
>>> a < b
True
>>> a < c
True
>>>

如果你想在多个线程中使用同一个队列,那么你需要增加适当的锁和信号量机制。 可以查看 12.3 小节的例子演示是怎样做的。

heapq 模块的官方文档有更详细的例子程序以及对于堆理论及其实现的详细说明。

以上就是Python实现一个优先级队列的方法的详细内容,更多关于Python实现优先级队列的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
Python 冒泡,选择,插入排序使用实例
Feb 05 Python
Python中用format函数格式化字符串的用法
Apr 08 Python
Python按行读取文件的实现方法【小文件和大文件读取】
Sep 19 Python
使用Django Form解决表单数据无法动态刷新的两种方法
Jul 14 Python
Python三级菜单的实例
Sep 13 Python
Python+tkinter模拟“记住我”自动登录实例代码
Jan 16 Python
Python3实现转换Image图片格式
Jun 21 Python
解决Python print 输出文本显示 gbk 编码错误问题
Jul 13 Python
selenium + python 获取table数据的示例讲解
Oct 13 Python
PyTorch中topk函数的用法详解
Jan 02 Python
Python astype(np.float)函数使用方法解析
Jun 08 Python
opencv 分类白天与夜景视频的方法
Jun 05 Python
django表单中的按钮获取数据的实例分析
Jul 31 #Python
pycharm中使用request和Pytest进行接口测试的方法
Jul 31 #Python
django创建css文件夹的具体方法
Jul 31 #Python
Selenium之模拟登录铁路12306的示例代码
Jul 31 #Python
python的flask框架难学吗
Jul 31 #Python
使用PyCharm安装pytest及requests的问题
Jul 31 #Python
django和flask哪个值得研究学习
Jul 31 #Python
You might like
PHP获取表单textarea数据中的换行问题
2010/09/10 PHP
PHP生成随机用户名和密码的实现代码
2013/02/27 PHP
php selectradio和checkbox默认选择的实现方法详解
2013/06/29 PHP
php实现多维数组中每个单元值(数字)翻倍的方法
2015/02/16 PHP
PHP 7的一些引人注目的新特性简单介绍
2015/11/08 PHP
PHP大文件切割上传功能实例分析
2019/07/01 PHP
半角全角相互转换的js函数
2009/10/16 Javascript
php析构函数的具体用法小结
2014/03/11 Javascript
js 显示base64编码的二进制流网页图片
2014/04/04 Javascript
浅谈javascript获取元素transform参数
2015/07/24 Javascript
jQuery 1.9.1源码分析系列(十五)动画处理之缓动动画核心Tween
2015/12/03 Javascript
jQuery form插件的使用之处理server返回的JSON, XML,HTML数据
2016/01/26 Javascript
Js遍历键值对形式对象或Map形式的方法
2016/08/08 Javascript
使用vue + less 实现简单换肤功能的示例
2018/02/21 Javascript
探秘vue-rx 2.0(推荐)
2018/09/21 Javascript
vue: WebStorm设置快速编译运行的方法
2018/10/18 Javascript
详解JS实现简单的时分秒倒计时代码
2019/04/25 Javascript
vue中的.$mount('#app')手动挂载操作
2020/09/02 Javascript
简单介绍Python中的几种数据类型
2016/01/02 Python
解决PyCharm中光标变粗的问题
2017/08/05 Python
python求最大连续子数组的和
2018/07/07 Python
详解如何为eclipse安装合适版本的python插件pydev
2018/11/04 Python
Python基于opencv调用摄像头获取个人图片的实现方法
2019/02/21 Python
对python中不同模块(函数、类、变量)的调用详解
2019/07/16 Python
PyTorch之图像和Tensor填充的实例
2019/08/18 Python
Tensorflow实现多GPU并行方式
2020/02/03 Python
Python随机数函数代码实例解析
2020/02/09 Python
一篇文章搞懂python的转义字符及用法
2020/09/03 Python
python3定位并识别图片验证码实现自动登录功能
2021/01/29 Python
EJB timer的种类
2014/10/28 面试题
电大自我鉴定范文
2013/10/01 职场文书
哈弗商学院毕业生求职信
2014/02/26 职场文书
服装店营销方案
2014/03/10 职场文书
优秀教师单行材料
2014/12/16 职场文书
详解nodejs内置模块
2021/05/06 NodeJs
Spring Security使用单点登录的权限功能
2022/04/03 Java/Android