- 前言
創(chuàng)作動機
- 什麼是 Heap (堆積)?
16 / \ / \ / \ / \ / \ / \ / \ 80 41 / \ / \ / \ / \ / \ / \ 32 82 95 35 / \ / \ / \ / \ 47 17 6 43 97 |
79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 67 91 3 17 / \ / \ / \ / \ 69 54 38 43 86 81 51 13 ↓ 把一棵完全二元樹轉(zhuǎn)成 max-heap 96 / \ / \ / \ / \ / \ / \ / \ 91 86 / \ / \ / \ / \ / \ / \ 69 85 81 51 / \ / \ / \ / \ 67 54 38 43 3 79 17 13 |
- 如何造 Heap?
79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 67 91 3 17 / \ / \ / \ / \ 69 54 38 43 86 81 85 13 85 > 17 所以互換 79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 67 91 3 85 / \ / \ / \ / \ 69 54 38 43 86 81 17 13 86 > 3 所以互換 79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 67 91 86 85 / \ / \ / \ / \ 69 54 38 43 3 81 17 13 91 比自己子節(jié)點都大,不動 79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 67 91 86 85 / \ / \ / \ / \ 69 54 38 43 3 81 17 13 69 > 67 所以互換 79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 69 91 86 85 / \ / \ / \ / \ 67 54 38 43 3 81 17 13 96 比自己子節(jié)點都大,不動 79 / \ / \ / \ / \ / \ / \ / \ 85 96 / \ / \ / \ / \ / \ / \ 69 91 86 85 / \ / \ / \ / \ 67 54 38 43 3 81 17 13 91 > 85 所以互換 79 / \ / \ / \ / \ / \ / \ / \ 91 96 / \ / \ / \ / \ / \ / \ 69 85 86 85 / \ / \ / \ / \ 67 54 38 43 3 81 17 13 96 > 79 所以互換 往下遞迴檢查 86 > 79 所以互換 往下遞迴檢查 81 > 79 所以互換 以到葉節(jié)點 直接回傳 96 / \ / \ / \ / \ / \ / \ / \ 91 86 / \ / \ / \ / \ / \ / \ 69 85 81 85 / \ / \ / \ / \ 67 54 38 43 3 79 17 13 |
- 造 Heap 的程式碼
/** * @param arr - The array that is going to be built as a heap. * @param n - The size of the array. * @param parent - The parent node index. */ void heapify(vector<int> &arr, int n, int parent) { int childLeft = parent*2 + 1; int childRight = parent*2 + 2; int greaterChild; if (childLeft >= n) { return; } if (childRight >= n || arr[childLeft] > arr[childRight]) { greaterChild = childLeft; } else { greaterChild = childRight; } if (arr[greaterChild] > arr[parent]) { swap(arr[greaterChild], arr[parent]); heapify(arr, n, greaterChild); } } /** * @param arr - The array that is going to be built as a heap. * @param n - The size of the array. */ void buildHeap(vector<int> &arr, int n) { for (int parent = (n-2)/2; parent >= 0; --parent) { heapify(arr, n, parent); } } |
- Heap sort (堆積排序)
96 / \ / \ / \ / \ / \ / \ / \ 91 86 / \ / \ / \ / \ / \ / \ 69 85 81 85 / \ / \ / \ / \ 67 54 38 43 3 79 17 13 13 96 互換 把 96 移到新陣列 並且對根節(jié)點 13 做 heapify 91 / \ / \ / \ / \ / \ / \ / \ 85 86 / \ / \ / \ / \ / \ / \ 69 43 81 85 / \ / \ / \ / \ 67 54 38 13 3 79 17 此時已排序的陣列: {96} 91 / \ / \ / \ / \ / \ / \ / \ 85 86 / \ / \ / \ / \ / \ / \ 69 43 81 85 / \ / \ / \ / \ 67 54 38 13 3 79 17 17 91 互換 把 91 移到新陣列 並且對根節(jié)點 17 做 heapify 86 /\ / \ / \ / \ / \ / \ / \ 85 85 / \ / \ / \ / \ / \ / \ 69 43 81 17 / \ / \ / \ / \ 67 54 38 13 3 79 此時已排序的陣列: {96, 91} 86 /\ / \ / \ / \ / \ / \ / \ 85 85 / \ / \ / \ / \ / \ / \ 69 43 81 17 / \ / \ / \ / \ 67 54 38 13 3 79 79 86 互換 把 86 移到新陣列 並且對根節(jié)點 79 做 heapify 85 /\ / \ / \ / \ / \ / \ / \ 85 81 / \ / \ / \ / \ / \ / \ 69 43 79 17 / \ / \ / \ / \ 67 54 38 13 3 此時已排序的陣列: {96, 91, 86} 85 /\ / \ / \ / \ / \ / \ / \ 85 81 / \ / \ / \ / \ / \ / \ 69 43 79 17 / \ / \ / \ / \ 67 54 38 13 3 3 85 互換 把 85 移到新陣列 並且對根節(jié)點 3 做 heapify 85 / \ / \ / \ / \ / \ / \ 69 81 /\ /\ / \ / \ / \ / \ 67 43 79 17 / \ / \ / \ / \ 3 54 38 13 此時已排序的陣列: {96, 91, 86, 85} 85 / \ / \ / \ / \ / \ / \ 69 81 /\ /\ / \ / \ / \ / \ 67 43 79 17 / \ / \ / \ / \ 3 54 38 13 13 85 互換 把 85 移到新陣列 並且對根節(jié)點 13 做 heapify 81 / \ / \ / \ / \ / \ / \ 69 79 /\ /\ / \ / \ / \ / \ 67 43 13 17 / \ / \ / \ / \ 3 54 38 此時已排序的陣列: {96, 91, 86, 85, 85} 81 / \ / \ / \ / \ / \ / \ 69 79 /\ /\ / \ / \ / \ / \ 67 43 13 17 / \ / \ / \ / \ 3 54 38 38 81 互換 把 81 移到新陣列 並且對根節(jié)點 38 做 heapify 79 /\ / \ / \ / \ / \ / \ 69 38 /\ /\ / \ / \ / \ / \ 67 43 13 17 / \ / \ / \ / \ 3 54 此時已排序的陣列: {96, 91, 86, 85, 85, 81} 79 /\ / \ / \ / \ / \ / \ 69 38 /\ /\ / \ / \ / \ / \ 67 43 13 17 / \ / \ / \ / \ 3 54 54 79 互換 把 79 移到新陣列 並且對根節(jié)點 54 做 heapify 69 /\ / \ / \ / \ / \ / \ 67 38 /\ /\ / \ / \ / \ / \ 54 43 13 17 / \ / \ / \ / \ 3 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79} 69 /\ / \ / \ / \ / \ / \ 67 38 /\ /\ / \ / \ / \ / \ 54 43 13 17 / \ / \ / \ / \ 3 3 69 互換 把 69 移到新陣列 並且對根節(jié)點 3 做 heapify 67 / \ / \ / \ 54 38 / \ / \ 3 43 13 17 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69} 67 / \ / \ / \ 54 38 / \ / \ 3 43 13 17 17 67 互換 把 67 移到新陣列 並且對根節(jié)點 17 做 heapify 54 / \ / \ / \ 43 38 / \ / \ 3 17 13 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67} 54 / \ / \ / \ 43 38 / \ / \ 3 17 13 13 54 互換 把 54 移到新陣列 並且對根節(jié)點 13 做 heapify 43 /\ / \ / \ 17 38 / \ / \ 3 13 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54} 43 /\ / \ / \ 17 38 / \ / \ 3 13 13 43 互換 把 43 移到新陣列 並且對根節(jié)點 13 做 heapify 38 /\ / \ / \ 17 13 / \ / \ 3 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54, 43} 38 /\ / \ / \ 17 13 / \ / \ 3 3 38 互換 把 38 移到新陣列 並且對根節(jié)點 3 做 heapify 17 / \ 3 13 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54, 43, 38} 17 / \ 3 13 13 17 互換 把 17 移到新陣列 並且對根節(jié)點 13 做 heapify 13 / \ 3 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54, 43, 38, 17} 13 / \ 3 3 13 互換 把 13 移到新陣列 並且對根節(jié)點 3 做 heapify 3 / \ 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54, 43, 38, 17, 13} 3 / \ 3 3 互換 把 3 移到新陣列 並且對根節(jié)點 3 做 heapify 此時已排序的陣列: {96, 91, 86, 85, 85, 81, 79, 69, 67, 54, 43, 38, 17, 13, 3} |
- 其他
- Heap sort 總體程式碼
/** * @param arr - The array that is going to be built as a heap. * @param n - The size of the array. * @param parent - The parent node index. */ void heapify(vector<int> &arr, int n, int parent) { int childLeft = parent*2 + 1; int childRight = parent*2 + 2; int greaterChild; if (childLeft >= n) { return; } if (childRight >= n || arr[childLeft] > arr[childRight]) { greaterChild = childLeft; } else { greaterChild = childRight; } if (arr[greaterChild] > arr[parent]) { swap(arr[greaterChild], arr[parent]); heapify(arr, n, greaterChild); } } /** * @param arr - The array that is going to be built as a heap. * @param n - The size of the array. */ void buildHeap(vector<int> &arr, int n) { for (int parent = (n-2)/2; parent >= 0; --parent) { heapify(arr, n, parent); } } /** * @param arr - The array that is going to be sorted. */ void heapSort(vector<int> &arr) { vector<int> sortedArr; int n = arr.size(); for (int i = 0; i < n; ++i) { print_tree(arr); swap(arr[0], arr[arr.size()-1]); sortedArr.push_back(arr[arr.size()-1]); arr.pop_back(); heapify(arr, arr.size(), 0); } arr = sortedArr; } int main() { const int n = 15; vector<int> arr = {79,85,96,67,91,3,17,69,54,38,43,86,81,85,13}; buildHeap(arr, n); heapSort(arr); return 0; } |