左傾堆,用於堆的快速合併。 規則: ① 節點的鍵值小於或等於它的左右子節點的鍵值。 ② 節點的左孩子的NPL >= 右孩子的NPL。 ③ 節點的NPL = 它的右孩子的NPL + 1。 測試文件 main.cpp: #include <iostream> #include "LeftistHeap. ...
左傾堆,用於堆的快速合併。
規則:
① 節點的鍵值小於或等於它的左右子節點的鍵值。
② 節點的左孩子的NPL >= 右孩子的NPL。
③ 節點的NPL = 它的右孩子的NPL + 1。
測試文件 main.cpp:
#include <iostream> #include "LeftistHeap.h" using std::cout; using std::endl; int main() { LeftistHeap<int> lh(LeftistHeap<int>::HeapType::MINIMEM); auto il = { 1,2,3,4,5,5,6,7,8,9 }; for (auto& x : il) lh.push(x); cout << "Element:\n\t"; lh.levelTraversal(); cout << endl << endl; cout << "Pop: " << lh.top() << endl << endl; lh.pop(); cout << "Element:\n\t"; lh.levelTraversal(); cout << endl; return 0; }
頭文件 "LeftistHeap.h":
#pragma once #ifndef __LEFTISTHEAP_H__ #define __LEFTISTHEAP_H__ #include "BinaryTreeOperations.h" template<typename _Ty> class LeftistHeap { struct Node { _Ty key; int NPL = 0; Node* left = nullptr; Node* right = nullptr; Node(const _Ty& _key) :key(_key) {} }; public: enum class HeapType :bool { MINIMEM = 0, MAXIMEM }; public: LeftistHeap() = default; LeftistHeap(HeapType _heapType) { heapType = _heapType; } ~LeftistHeap() { BTO::clear(root); size_n = 0; } void clear() noexcept { BTO::clear(root); size_n = 0; } void preorderTraversal() { BTO::preorderTraversal(root, drawData); } void inorderTraversal() { BTO::inorderTraversal(root, drawData); } void postorderTraversal() { BTO::postorderTraversal(root, drawData); } void iterativePreorderTraversal() { BTO::iterativePreorderTraversal(root, drawData); } void iterativeInorderTraversal() { BTO::iterativeInorderTraversal(root, drawData); } void iterativePostorderTraversal() { BTO::iterativePostorderTraversal(root, drawData); } void levelTraversal() { BTO::levelTraversal(root, drawData); } size_t size() const { return size_n; } void pop(); _Ty& top() const; void push(const _Ty&); void merge(LeftistHeap<_Ty>&); private: static void drawData(const Node* _node) { std::cout << _node->key << " "; } bool compare(const _Ty& _a, const _Ty& _b) { return (heapType == HeapType::MAXIMEM) ? (_a > _b) : (_a < _b); } Node* merge(Node*&, Node*&); private: Node* root = nullptr; size_t size_n = 0; HeapType heapType = HeapType::MAXIMEM; }; template<typename _Ty> void LeftistHeap<_Ty>::pop() { if (root == nullptr) throw std::exception("LeftistHeap is empty!"); Node* leftT = root->left; Node* rightT = root->right; delete root; root = merge(leftT, rightT); --size_n; } template<typename _Ty> _Ty& LeftistHeap<_Ty>::top() const { if (root == nullptr) throw std::exception("LeftistHeap is empty!"); return root->key; } template<typename _Ty> void LeftistHeap<_Ty>::push(const _Ty& _key) { Node* temp = new Node(_key); root = merge(root, temp); temp = nullptr; ++size_n; } template<typename _Ty> void LeftistHeap<_Ty>::merge(LeftistHeap<_Ty>& _lh) { if (heapType != _lh.heapType) throw std::exception("Bad heapType"); root = merge(root, _lh.root); _lh.root = nullptr; size_n += _lh.size_n; _lh.size_n = 0; } template<typename _Ty> typename LeftistHeap<_Ty>::Node* LeftistHeap<_Ty>::merge(Node*& _n1, Node*& _n2) { if (_n1 == nullptr && _n2 == nullptr) return nullptr; else if (_n1 == nullptr) return _n2; else if (_n2 == nullptr) return _n1; if (!compare(_n1->key, _n2->key)) std::swap(_n1, _n2); _n1->right = merge(_n1->right, _n2); if (_n1->left == nullptr || _n1->left->NPL < _n1->right->NPL) std::swap(_n1->left, _n1->right); if (_n1->right == nullptr || _n1->left == nullptr) _n1->NPL = 0; else _n1->NPL = _n1->right->NPL + 1; return _n1; } #endif // !__LEFTISTHEAP_H__