179 lines
3.9 KiB
Python
179 lines
3.9 KiB
Python
|
#!/usr/bin/python
|
||
|
# -*- coding: UTF-8 -*-
|
||
|
|
||
|
import math
|
||
|
import random
|
||
|
|
||
|
|
||
|
class Heap:
|
||
|
def __init__(self, nums=None, capacity=100):
|
||
|
self._data = []
|
||
|
self._capacity = capacity
|
||
|
if type(nums) == list and len(nums) <= self._capacity:
|
||
|
for n in nums:
|
||
|
assert type(n) is int
|
||
|
self._data.append(n)
|
||
|
self._length = len(self._data)
|
||
|
self._heapify()
|
||
|
|
||
|
def _heapify(self):
|
||
|
if self._length <= 1:
|
||
|
return
|
||
|
|
||
|
# idx of the Last Parent node
|
||
|
lp = (self._length - 2) // 2
|
||
|
|
||
|
for i in range(lp, -1, -1):
|
||
|
self._heap_down(i)
|
||
|
|
||
|
def _heap_down(self, idx):
|
||
|
pass
|
||
|
|
||
|
def insert(self, num):
|
||
|
pass
|
||
|
|
||
|
def get_top(self):
|
||
|
if self._length <= 0:
|
||
|
return None
|
||
|
return self._data[0]
|
||
|
|
||
|
def remove_top(self):
|
||
|
if self._length <= 0:
|
||
|
return None
|
||
|
|
||
|
self._data[0], self._data[-1] = self._data[-1], self._data[0]
|
||
|
ret = self._data.pop()
|
||
|
self._length -= 1
|
||
|
self._heap_down(0)
|
||
|
|
||
|
return ret
|
||
|
|
||
|
def get_data(self):
|
||
|
return self._data
|
||
|
|
||
|
def get_length(self):
|
||
|
return self._length
|
||
|
|
||
|
@staticmethod
|
||
|
def _draw_heap(data):
|
||
|
"""
|
||
|
格式化打印
|
||
|
:param data:
|
||
|
:return:
|
||
|
"""
|
||
|
length = len(data)
|
||
|
|
||
|
if length == 0:
|
||
|
return 'empty heap'
|
||
|
|
||
|
ret = ''
|
||
|
for i, n in enumerate(data):
|
||
|
ret += str(n)
|
||
|
# 每行最后一个换行
|
||
|
if i == 2 ** int(math.log(i + 1, 2) + 1) - 2 or i == len(data) - 1:
|
||
|
ret += '\n'
|
||
|
else:
|
||
|
ret += ', '
|
||
|
|
||
|
return ret
|
||
|
|
||
|
def __repr__(self):
|
||
|
return self._draw_heap(self._data)
|
||
|
|
||
|
|
||
|
class MaxHeap(Heap):
|
||
|
def _heap_down(self, idx):
|
||
|
if self._length <= 1:
|
||
|
return
|
||
|
|
||
|
lp = (self._length - 2) // 2
|
||
|
|
||
|
while idx <= lp:
|
||
|
lc = 2 * idx + 1
|
||
|
rc = lc + 1
|
||
|
|
||
|
if rc <= self._length-1:
|
||
|
tmp = lc if self._data[lc] > self._data[rc] else rc
|
||
|
else:
|
||
|
tmp = lc
|
||
|
|
||
|
if self._data[tmp] > self._data[idx]:
|
||
|
self._data[tmp], self._data[idx] = self._data[idx], self._data[tmp]
|
||
|
idx = tmp
|
||
|
else:
|
||
|
break
|
||
|
|
||
|
def insert(self, num):
|
||
|
if self._length >= self._capacity:
|
||
|
return False
|
||
|
|
||
|
self._data.append(num)
|
||
|
self._length += 1
|
||
|
|
||
|
nn = self._length - 1
|
||
|
while nn > 0:
|
||
|
p = (nn-1) // 2
|
||
|
|
||
|
if self._data[nn] > self._data[p]:
|
||
|
self._data[nn], self._data[p] = self._data[p], self._data[nn]
|
||
|
nn = p
|
||
|
else:
|
||
|
break
|
||
|
|
||
|
return True
|
||
|
|
||
|
|
||
|
class MinHeap(Heap):
|
||
|
def _heap_down(self, idx):
|
||
|
if self._length <= 1:
|
||
|
return
|
||
|
|
||
|
lp = (self._length - 2) // 2
|
||
|
|
||
|
while idx <= lp:
|
||
|
lc = 2 * idx + 1
|
||
|
rc = lc + 1
|
||
|
|
||
|
if rc <= self._length-1:
|
||
|
tmp = lc if self._data[lc] < self._data[rc] else rc
|
||
|
else:
|
||
|
tmp = lc
|
||
|
|
||
|
if self._data[tmp] < self._data[idx]:
|
||
|
self._data[tmp], self._data[idx] = self._data[idx], self._data[tmp]
|
||
|
idx = tmp
|
||
|
else:
|
||
|
break
|
||
|
|
||
|
def insert(self, num):
|
||
|
if self._length >= self._capacity:
|
||
|
return False
|
||
|
|
||
|
self._data.append(num)
|
||
|
self._length += 1
|
||
|
|
||
|
nn = self._length - 1
|
||
|
while nn > 0:
|
||
|
p = (nn-1) // 2
|
||
|
|
||
|
if self._data[nn] < self._data[p]:
|
||
|
self._data[nn], self._data[p] = self._data[p], self._data[nn]
|
||
|
nn = p
|
||
|
else:
|
||
|
break
|
||
|
|
||
|
return True
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
nums = list(range(10))
|
||
|
random.shuffle(nums)
|
||
|
|
||
|
max_h = MaxHeap(nums)
|
||
|
print('--- max heap ---')
|
||
|
print(max_h)
|
||
|
|
||
|
print('--- min heap ---')
|
||
|
min_h = MinHeap(nums)
|
||
|
print(min_h)
|