graphlib
--- 操作類似圖的結(jié)構(gòu)的功能?
源代碼: Lib/graphlib.py
- class graphlib.TopologicalSorter(graph=None)?
提供以拓?fù)浞绞綄晒9?jié)點(diǎn)的圖進(jìn)行排序的功能。
拓?fù)渑判蚴侵笀D中頂點(diǎn)的線性排序,使得對于每條從頂點(diǎn) u 到頂點(diǎn) v 的有向邊 u -> v,頂點(diǎn) u 都排在頂點(diǎn) v 之前。 例如,圖的頂點(diǎn)可以代表要執(zhí)行的任務(wù),而邊代表某一個任務(wù)必須在另一個任務(wù)之前執(zhí)行的約束條件;在這個例子中,拓?fù)渑判蛑皇侨蝿?wù)的有效序列。 完全拓?fù)渑判?當(dāng)且僅當(dāng)圖不包含有向環(huán),也就是說為有向無環(huán)圖時,完全拓?fù)渑判虿攀强赡艿摹?/p>
如果提供了可選的 graph 參數(shù)則它必須為一個表示有向無環(huán)圖的字典,其中的鍵為節(jié)點(diǎn)而值為包含圖中該節(jié)點(diǎn)的所有上級節(jié)點(diǎn)(即具有指向鍵中的值的邊的節(jié)點(diǎn))的可迭代對象。 額外的節(jié)點(diǎn)可以使用
add()
方法添加到圖中。在通常情況下,對給定的圖執(zhí)行排序所需的步驟如下:
通過可選的初始圖創(chuàng)建一個
TopologicalSorter
的實(shí)例。添加額外的節(jié)點(diǎn)到圖中。
在圖上調(diào)用
prepare()
。當(dāng)
is_active()
為True
時,迭代get_ready()
所返回的節(jié)點(diǎn)并加以處理。 完成處理后在每個節(jié)點(diǎn)上調(diào)用done()
。
在只需要對圖中的節(jié)點(diǎn)進(jìn)行立即排序并且不涉及并行性的情況下,可以直接使用便捷方法
TopologicalSorter.static_order()
:>>> graph = {"D": {"B", "C"}, "C": {"A"}, "B": {"A"}} >>> ts = TopologicalSorter(graph) >>> tuple(ts.static_order()) ('A', 'C', 'B', 'D')
這個類被設(shè)計用來在節(jié)點(diǎn)就緒時方便地支持對其并行處理。 例如:
topological_sorter = TopologicalSorter() # Add nodes to 'topological_sorter'... topological_sorter.prepare() while topological_sorter.is_active(): for node in topological_sorter.get_ready(): # Worker threads or processes take nodes to work on off the # 'task_queue' queue. task_queue.put(node) # When the work for a node is done, workers put the node in # 'finalized_tasks_queue' so we can get more nodes to work on. # The definition of 'is_active()' guarantees that, at this point, at # least one node has been placed on 'task_queue' that hasn't yet # been passed to 'done()', so this blocking 'get()' must (eventually) # succeed. After calling 'done()', we loop back to call 'get_ready()' # again, so put newly freed nodes on 'task_queue' as soon as # logically possible. node = finalized_tasks_queue.get() topological_sorter.done(node)
- add(node, *predecessors)?
將一個新節(jié)點(diǎn)及其上級節(jié)點(diǎn)添加到圖中。 node 以及 predecessors 中的所有元素都必須為可哈希對象。
如果附帶相同的節(jié)點(diǎn)參數(shù)多次調(diào)用,則依賴項的集合將為所有被傳入依賴項的并集。
可以添加不帶依賴項的節(jié)點(diǎn) (即不提供 predecessors) 或者重復(fù)提供依賴項。 如果有先前未提供的節(jié)點(diǎn)包含在 predecessors 中則它將被自動添加到圖中并且不帶自己的上級節(jié)點(diǎn)。
如果在
prepare()
之后被調(diào)用則會引發(fā)ValueError
。
- prepare()?
將圖標(biāo)記為已完成并檢查圖中是否存在環(huán)。 如何檢測到任何環(huán),則將引發(fā)
CycleError
,但get_ready()
仍可被用來獲取盡可能多的節(jié)點(diǎn)直到環(huán)阻塞了操作過程。 在調(diào)用此函數(shù)后,圖將無法再修改,因此不能再使用add()
添加更多的節(jié)點(diǎn)。
- is_active()?
如果可以取得更多進(jìn)展則返回
True
,否則返回False
。 如果環(huán)沒有阻塞操作,并且還存在尚未被TopologicalSorter.get_ready()
返回的已就緒節(jié)點(diǎn)或者已標(biāo)記為TopologicalSorter.done()
的節(jié)點(diǎn)數(shù)量少于已被TopologicalSorter.get_ready()
所返回的節(jié)點(diǎn)數(shù)量則還可以取得進(jìn)展。該類的
__bool__()
方法要使用此函數(shù),因此除了:if ts.is_active(): ...
可能會簡單地執(zhí)行:
if ts: ...
如果之前未調(diào)用
prepare()
就調(diào)用此函數(shù)則會引發(fā)ValueError
。
- done(*nodes)?
將
TopologicalSorter.get_ready()
所返回的節(jié)點(diǎn)集合標(biāo)記為已處理,解除對 nodes 中每個節(jié)點(diǎn)的后續(xù)節(jié)點(diǎn)的阻塞以便在將來通過對TopologicalSorter.get_ready()
的調(diào)用來返回它們。如果 nodes 中的任何節(jié)點(diǎn)已經(jīng)被之前對該方法的調(diào)用標(biāo)記為已處理或者如果未通過使用
TopologicalSorter.add()
將一個節(jié)點(diǎn)添加到圖中,如果未調(diào)用prepare()
即調(diào)用此方法或者如果節(jié)點(diǎn)尚未被get_ready()
所返回則將引發(fā)ValueError
。
- get_ready()?
返回由所有已就緒節(jié)點(diǎn)組成的
tuple
。 初始狀態(tài)下它將返回所有不帶上級節(jié)點(diǎn)的節(jié)點(diǎn),并且一旦通過調(diào)用TopologicalSorter.done()
將它們標(biāo)記為已處理,之后的調(diào)用將返回所有上級節(jié)點(diǎn)已被處理的新節(jié)點(diǎn)。 一旦無法再取得進(jìn)展,則會返回空元組。如果之前未調(diào)用
prepare()
就調(diào)用此函數(shù)則會引發(fā)ValueError
。
- static_order()?
返回一個迭代器,它將按照拓?fù)漤樞騺淼泄?jié)點(diǎn)。 當(dāng)使用此方法時,
prepare()
和done()
不應(yīng)被調(diào)用。 此方法等價于:def static_order(self): self.prepare() while self.is_active(): node_group = self.get_ready() yield from node_group self.done(*node_group)
所返回的特定順序可能取決于條目被插入圖中的順序。 例如:
>>> ts = TopologicalSorter() >>> ts.add(3, 2, 1) >>> ts.add(1, 0) >>> print([*ts.static_order()]) [2, 0, 1, 3] >>> ts2 = TopologicalSorter() >>> ts2.add(1, 0) >>> ts2.add(3, 2, 1) >>> print([*ts2.static_order()]) [0, 2, 1, 3]
這是由于實(shí)際上 "0" 和 "2" 在圖中的級別相同(它們將在對
get_ready()
的同一次調(diào)用中被返回) 并且它們之間的順序是由插入順序決定的。如果檢測到任何環(huán),則將引發(fā)
CycleError
。
3.9 新版功能.
異常?
graphlib
模塊定義了以下異常類:
- exception graphlib.CycleError?
ValueError
的子類,當(dāng)特定的圖中存在環(huán)時將由TopologicalSorter.prepare()
引發(fā)。 如果存在多個環(huán),則將只報告其中一個未定義的選項并將其包括在異常中。檢測到的環(huán)可以通過異常實(shí)例的
args
屬性的第二個元素來訪問,它由一個節(jié)點(diǎn)列表組成,其中的每個節(jié)點(diǎn)在圖中都是列表中下一個節(jié)點(diǎn)的直接上級節(jié)點(diǎn)。 在報告的列表中,開頭和末尾的節(jié)點(diǎn)將是同一對象,以表明它是一個環(huán)。