每週一個 Python 模塊 | heapq

專欄地址:每週一個 Python 模塊html

heapq 實現了適用於 Python 列表的最小堆排序算法。python

堆是一個樹狀的數據結構,其中的子節點與父節點屬於排序關係。能夠使用列表或數組來表示二進制堆,使得元素 N 的子元素位於 2 * N + 1 和 2 * N + 2 的位置(對於從零開始的索引)。這種佈局使得能夠在適當的位置從新排列堆,所以在添加或刪除數據時無需從新分配內存。git

max-heap 確保父級大於或等於其子級。min-heap 要求父項小於或等於其子級。Python 的heapq模塊實現了一個 min-heap。github

示例數據

本節中的示例使用數據heapq_heapdata.py算法

# heapq_heapdata.py 
# This data was generated with the random module.

data = [19, 9, 4, 10, 11]
複製代碼

堆輸出使用打印heapq_showtree.pyapi

# heapq_showtree.py 
import math
from io import StringIO


def show_tree(tree, total_width=36, fill=' '):
    """Pretty-print a tree."""
    output = StringIO()
    last_row = -1
    for i, n in enumerate(tree):
        if i:
            row = int(math.floor(math.log(i + 1, 2)))
        else:
            row = 0
        if row != last_row:
            output.write('\n')
        columns = 2 ** row
        col_width = int(math.floor(total_width / columns))
        output.write(str(n).center(col_width, fill))
        last_row = row
    print(output.getvalue())
    print('-' * total_width)
    print()
複製代碼

建立堆

建立堆有兩種基本方法:heappush()heapify()數組

import heapq
from heapq_showtree import show_tree
from heapq_heapdata import data

heap = []
print('random :', data)
print()

for n in data:
    print('add {:>3}:'.format(n))
    heapq.heappush(heap, n)
    show_tree(heap)
    
# output
# random : [19, 9, 4, 10, 11]
# 
# add 19:
# 
# 19
# ------------------------------------
# 
# add 9:
# 
# 9
# 19
# ------------------------------------
# 
# add 4:
# 
# 4
# 19 9
# ------------------------------------
# 
# add 10:
# 
# 4
# 10 9
# 19
# ------------------------------------
# 
# add 11:
# 
# 4
# 10 9
# 19 11
# ------------------------------------
複製代碼

當使用heappush()時,當新元素添加時,堆得順序被保持了。數據結構

若是數據已經在內存中,則使用 heapify() 來更有效地從新排列列表中的元素。app

import heapq
from heapq_showtree import show_tree
from heapq_heapdata import data

print('random :', data)
heapq.heapify(data)
print('heapified :')
show_tree(data)

# output
# random : [19, 9, 4, 10, 11]
# heapified :
# 
# 4
# 9 19
# 10 11
# ------------------------------------
複製代碼

訪問堆的內容

正確建立堆後,使用heappop()刪除具備最小值的元素。dom

import heapq
from heapq_showtree import show_tree
from heapq_heapdata import data

print('random :', data)
heapq.heapify(data)
print('heapified :')
show_tree(data)
print()

for i in range(2):
    smallest = heapq.heappop(data)
    print('pop {:>3}:'.format(smallest))
    show_tree(data)
    
# output
# random : [19, 9, 4, 10, 11]
# heapified :
# 
# 4
# 9 19
# 10 11
# ------------------------------------
# 
# 
# pop 4:
# 
# 9
# 10 19
# 11
# ------------------------------------
# 
# pop 9:
# 
# 10
# 11 19
# ------------------------------------
複製代碼

在這個例子中,使用 heapify()heappop() 進行排序。

要刪除現有元素,並在一次操做中用新值替換它們,使用heapreplace()

import heapq
from heapq_showtree import show_tree
from heapq_heapdata import data

heapq.heapify(data)
print('start:')
show_tree(data)

for n in [0, 13]:
    smallest = heapq.heapreplace(data, n)
    print('replace {:>2} with {:>2}:'.format(smallest, n))
    show_tree(data)
    
# output
# start:
# 
# 4
# 9 19
# 10 11
# ------------------------------------
# 
# replace 4 with 0:
# 
# 0
# 9 19
# 10 11
# ------------------------------------
# 
# replace 0 with 13:
# 
# 9
# 10 19
# 13 11
# ------------------------------------
複製代碼

替換元素能夠維護固定大小的堆,例如按優先級排序的 jobs 隊列。

堆的數據極值

heapq 還包括兩個函數來檢查 iterable 並找到它包含的最大或最小值的範圍。

import heapq
from heapq_heapdata import data

print('all :', data)
print('3 largest :', heapq.nlargest(3, data))
print('from sort :', list(reversed(sorted(data)[-3:])))
print('3 smallest:', heapq.nsmallest(3, data))
print('from sort :', sorted(data)[:3])

# output
# all : [19, 9, 4, 10, 11]
# 3 largest : [19, 11, 10]
# from sort : [19, 11, 10]
# 3 smallest: [4, 9, 10]
# from sort : [4, 9, 10]
複製代碼

使用nlargest()nsmallest()僅對 n > 1 的相對較小的值有效,但在少數狀況下仍然能夠派上用場。

有效地合併排序序列

將幾個排序的序列組合成一個新序列對於小數據集來講很容易。

list(sorted(itertools.chain(*data)))
複製代碼

對於較大的數據集,將會佔用大量內存。不是對整個組合序列進行排序,而是使用 merge() 一次生成一個新序列。

import heapq
import random


random.seed(2016)

data = []
for i in range(4):
    new_data = list(random.sample(range(1, 101), 5))
    new_data.sort()
    data.append(new_data)

for i, d in enumerate(data):
    print('{}: {}'.format(i, d))

print('\nMerged:')
for i in heapq.merge(*data):
    print(i, end=' ')
print()

# output
# 0: [33, 58, 71, 88, 95]
# 1: [10, 11, 17, 38, 91]
# 2: [13, 18, 39, 61, 63]
# 3: [20, 27, 31, 42, 45]
# 
# Merged:
# 10 11 13 17 18 20 27 31 33 38 39 42 45 58 61 63 71 88 91 95
複製代碼

由於merge()使用堆的實現,它根據被合併的序列元素個數消耗內存,而不是全部序列中的元素個數。

相關文檔:

pymotw.com/3/heapq/ind…

相關文章
相關標籤/搜索