Python實(shí)現(xiàn)一個(gè)優(yōu)先級(jí)隊(duì)列的方法
問(wèn)題
怎樣實(shí)現(xiàn)一個(gè)按優(yōu)先級(jí)排序的隊(duì)列? 并且在這個(gè)隊(duì)列上面每次 pop 操作總是返回優(yōu)先級(jí)最高的那個(gè)元素
解決方案
下面的類(lèi)利用 heapq
模塊實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的優(yōu)先級(jí)隊(duì)列:
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') >>>
仔細(xì)觀察可以發(fā)現(xiàn),第一個(gè) pop()
操作返回優(yōu)先級(jí)最高的元素。 另外注意到如果兩個(gè)有著相同優(yōu)先級(jí)的元素( foo
和 grok
),pop 操作按照它們被插入到隊(duì)列的順序返回的。
討論
這一小節(jié)我們主要關(guān)注 heapq
模塊的使用。 函數(shù) heapq.heappush()
和 heapq.heappop()
分別在隊(duì)列 _queue
上插入和刪除第一個(gè)元素, 并且隊(duì)列 _queue
保證第一個(gè)元素?fù)碛凶罡邇?yōu)先級(jí)( 1.4 節(jié)已經(jīng)討論過(guò)這個(gè)問(wèn)題)。 heappop()
函數(shù)總是返回”最小的”的元素,這就是保證隊(duì)列pop操作返回正確元素的關(guān)鍵。 另外,由于 push 和 pop 操作時(shí)間復(fù)雜度為 O(log N),其中 N 是堆的大小,因此就算是 N 很大的時(shí)候它們運(yùn)行速度也依舊很快。
在上面代碼中,隊(duì)列包含了一個(gè) (-priority, index, item)
的元組。 優(yōu)先級(jí)為負(fù)數(shù)的目的是使得元素按照優(yōu)先級(jí)從高到低排序。 這個(gè)跟普通的按優(yōu)先級(jí)從低到高排序的堆排序恰巧相反。
index
變量的作用是保證同等優(yōu)先級(jí)元素的正確排序。 通過(guò)保存一個(gè)不斷增加的 index
下標(biāo)變量,可以確保元素按照它們插入的順序排序。 而且, index
變量也在相同優(yōu)先級(jí)元素比較的時(shí)候起到重要作用。
為了闡明這些,先假定 Item
實(shí)例是不支持排序的:
>>> 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)
,只要兩個(gè)元素的優(yōu)先級(jí)不同就能比較。 但是如果兩個(gè)元素優(yōu)先級(jí)一樣的話(huà),那么比較操作就會(huì)跟之前一樣出錯(cuò):
>>> 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() >>>
通過(guò)引入另外的 index
變量組成三元組 (priority, index, item)
,就能很好的避免上面的錯(cuò)誤, 因?yàn)椴豢赡苡袃蓚€(gè)元素有相同的 index
值。Python 在做元組比較時(shí)候,如果前面的比較已經(jīng)可以確定結(jié)果了, 后面的比較操作就不會(huì)發(fā)生了:
>>> a = (1, 0, Item('foo')) >>> b = (5, 1, Item('bar')) >>> c = (1, 2, Item('grok')) >>> a < b True >>> a < c True >>>
如果你想在多個(gè)線(xiàn)程中使用同一個(gè)隊(duì)列,那么你需要增加適當(dāng)?shù)逆i和信號(hào)量機(jī)制。 可以查看 12.3 小節(jié)的例子演示是怎樣做的。
heapq
模塊的官方文檔有更詳細(xì)的例子程序以及對(duì)于堆理論及其實(shí)現(xiàn)的詳細(xì)說(shuō)明。
以上就是Python實(shí)現(xiàn)一個(gè)優(yōu)先級(jí)隊(duì)列的方法的詳細(xì)內(nèi)容,更多關(guān)于Python實(shí)現(xiàn)優(yōu)先級(jí)隊(duì)列的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- python線(xiàn)程優(yōu)先級(jí)隊(duì)列知識(shí)點(diǎn)總結(jié)
- 如何通過(guò)Python實(shí)現(xiàn)RabbitMQ延遲隊(duì)列
- python分布式爬蟲(chóng)中消息隊(duì)列知識(shí)點(diǎn)詳解
- Python通過(guò)隊(duì)列來(lái)實(shí)現(xiàn)進(jìn)程間通信的示例
- Python collections.deque雙邊隊(duì)列原理詳解
- 基于python實(shí)現(xiàn)操作redis及消息隊(duì)列
- Python Celery異步任務(wù)隊(duì)列使用方法解析
- Python如何使用隊(duì)列方式實(shí)現(xiàn)多線(xiàn)程爬蟲(chóng)
- Python多線(xiàn)程通信queue隊(duì)列用法實(shí)例分析
- python3 deque 雙向隊(duì)列創(chuàng)建與使用方法分析
- Python實(shí)現(xiàn)隊(duì)列的方法示例小結(jié)【數(shù)組,鏈表】
- 詳解python數(shù)據(jù)結(jié)構(gòu)之隊(duì)列Queue
相關(guān)文章
Python網(wǎng)絡(luò)編程使用select實(shí)現(xiàn)socket全雙工異步通信功能示例
這篇文章主要介紹了Python網(wǎng)絡(luò)編程使用select實(shí)現(xiàn)socket全雙工異步通信功能,簡(jiǎn)單說(shuō)明了select模塊的功能及socket全雙工異步通信功能的相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2018-04-04python實(shí)現(xiàn)telnet客戶(hù)端的方法
這篇文章主要介紹了python實(shí)現(xiàn)telnet客戶(hù)端的方法,分析了Python中telnetlib模塊實(shí)現(xiàn)telnet操作的方法,并實(shí)例敘述了Telnet客戶(hù)端的實(shí)現(xiàn)技巧,需要的朋友可以參考下2015-04-04Python實(shí)現(xiàn)不一樣的猜數(shù)字游戲的示例代碼
大家知道“猜數(shù)字”這個(gè)游戲嗎?顧名思義就是一個(gè)人想一個(gè)數(shù)字,另一個(gè)人猜。本文就來(lái)用Python實(shí)現(xiàn)一款不一樣的猜數(shù)字游戲,感興趣的可以了解一下2023-02-02python中使用smtplib和email模塊發(fā)送郵件實(shí)例
python腳本發(fā)郵件,一般會(huì)用到smtplib和email這兩個(gè)模塊。看看該模塊怎么使用,先看smtplib模塊。 smtplib模塊定義了一個(gè)簡(jiǎn)單的SMTP客戶(hù)端,可以用來(lái)在互聯(lián)網(wǎng)上發(fā)送郵件2014-04-04Python模塊_PyLibTiff讀取tif文件的實(shí)例
今天小編就為大家分享一篇Python模塊_PyLibTiff讀取tif文件的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-01-01python3中_from...import...與import?...之間的區(qū)別詳解(包/模塊)
Python編碼第一步是導(dǎo)入模塊,有時(shí)候用import?***有時(shí)候用from...import,下面這篇文章主要給大家介紹了關(guān)于python3中_from...import...與import?...之間區(qū)別的相關(guān)資料,需要的朋友可以參考下2022-08-08python如何實(shí)現(xiàn)單向鏈表及單向鏈表的反轉(zhuǎn)
這篇文章主要介紹了python如何實(shí)現(xiàn)單向鏈表及單向鏈表的反轉(zhuǎn),幫助大家更好的理解和學(xué)習(xí)使用python,感興趣的朋友可以了解下2021-03-03