AVL樹node
引用:https://www.cnblogs.com/skywang12345/ios
詳解之後再補充。。。git
代碼github
這是Qt Creator建立的工程編程
其他代碼今後處獲取:https://github.com/Duacai/Data-Structure-and-Algorithms/tree/master/Data-Structure/Tree/AVLTree編程語言
AVLTree.h函數
1 #ifndef AVLTREE_H 2 #define AVLTREE_H 3 4 #include <iostream> 5 #include <queue> 6 #include <iomanip> 7 8 using namespace std; 9 10 namespace Viclib 11 { 12 13 template <typename T> 14 class AVLTreeNode 15 { 16 public: 17 T key; 18 uint16_t height; 19 AVLTreeNode* left; 20 AVLTreeNode* right; 21 22 AVLTreeNode(T v, AVLTreeNode* l, AVLTreeNode* r) : 23 key(v), height(0), left(l), right(r) {} 24 }; 25 26 template <typename T> 27 class AVLTree 28 { 29 private: 30 AVLTreeNode<T>* mRoot; 31 uint64_t mCount; 32 33 void preOrder(AVLTreeNode<T>* tree) const; 34 void inOrder(AVLTreeNode<T>* tree) const; 35 void postOrder(AVLTreeNode<T>* tree) const; 36 37 void levelOrder(AVLTreeNode<T>* tree) const; 38 39 AVLTreeNode<T>* search(AVLTreeNode<T>* tree, T key) const; 40 AVLTreeNode<T>* iterativeSearch(AVLTreeNode<T>* tree, T key) const; 41 42 AVLTreeNode<T>* minimum(AVLTreeNode<T>* tree) const; 43 AVLTreeNode<T>* maximum(AVLTreeNode<T>* tree) const; 44 45 AVLTreeNode<T>* llRotation(AVLTreeNode<T>* tree) const; 46 AVLTreeNode<T>* rrRotation(AVLTreeNode<T>* tree) const; 47 AVLTreeNode<T>* lrRotation(AVLTreeNode<T>* tree) const; 48 AVLTreeNode<T>* rlRotation(AVLTreeNode<T>* tree) const; 49 50 AVLTreeNode<T>* insert(AVLTreeNode<T>* &tree, T key); 51 AVLTreeNode<T>* remove(AVLTreeNode<T>* &tree, AVLTreeNode<T>* del); 52 53 void printGraph(const void* mRoot, uint16_t m_keyStrLen) const; 54 void printTree(AVLTreeNode<T> const* const tree, bool firstNode) const; 55 56 void destroy(AVLTreeNode<T>* &tree) const; 57 uint16_t height(AVLTreeNode<T>* tree) const; 58 59 public: 60 AVLTree(); 61 virtual ~AVLTree(); 62 63 void preOrder() const; 64 void inOrder() const; 65 void postOrder() const; 66 67 void levelOrder() const; 68 69 AVLTreeNode<T>* search(T key) const; // 遞歸版 70 AVLTreeNode<T>* iterativeSearch(T key) const; // 非遞歸版 71 72 T minimum() const; 73 T maximum() const; 74 75 void insert(T key); 76 bool remove(T key); 77 78 void print() const; // 打印結點關係,誰是誰的結點 79 void printGraph(uint16_t keyStrLen) const; // 以圖形樹方式打印關係 80 void printTree() const; 81 82 void destroy(); 83 uint16_t height() const; 84 85 uint64_t getCount() const; 86 bool rootIsNullptr() const; 87 88 T getRootKey() const; 89 }; 90 91 template <typename T> 92 AVLTree<T>::AVLTree() : mRoot(nullptr), mCount(0) 93 { 94 } 95 96 template <typename T> 97 void AVLTree<T>::preOrder(AVLTreeNode<T>* tree) const // 前序遍歷 98 { 99 if ( tree != nullptr ) 100 { 101 cout << tree->key << " " << flush; 102 preOrder(tree->left); 103 preOrder(tree->right); 104 } 105 } 106 107 template <typename T> 108 void AVLTree<T>::preOrder() const 109 { 110 preOrder(mRoot); 111 cout << endl; 112 } 113 114 template <typename T> 115 void AVLTree<T>::inOrder(AVLTreeNode<T>* tree) const // 中序遍歷 116 { 117 if ( tree != nullptr ) 118 { 119 inOrder(tree->left); 120 cout << tree->key << " " << flush; 121 inOrder(tree->right); 122 } 123 } 124 125 template <typename T> 126 void AVLTree<T>::inOrder() const 127 { 128 inOrder(mRoot); 129 cout << endl; 130 } 131 132 template <typename T> 133 void AVLTree<T>::postOrder(AVLTreeNode<T>* tree) const // 後續遍歷 134 { 135 if ( tree != nullptr ) 136 { 137 postOrder(tree->left); 138 postOrder(tree->right); 139 cout << tree->key << " " << flush; 140 } 141 } 142 143 template <typename T> 144 void AVLTree<T>::postOrder() const 145 { 146 postOrder(mRoot); 147 cout << endl; 148 } 149 150 template <typename T> 151 void AVLTree<T>::levelOrder(AVLTreeNode<T>* tree) const // 廣度優先 152 { 153 if ( tree != nullptr ) 154 { 155 queue<AVLTreeNode<T>*> tmp; 156 tmp.push(tree); 157 158 while( tmp.size() > 0 ) 159 { 160 AVLTreeNode<T>* t = tmp.front(); 161 162 if ( t->left != nullptr ) 163 tmp.push(t->left); 164 165 if ( t->right != nullptr ) 166 tmp.push(t->right); 167 168 tmp.pop(); 169 170 cout << t->key << " " << flush; 171 } 172 } 173 } 174 175 template <typename T> 176 void AVLTree<T>::levelOrder() const 177 { 178 levelOrder(mRoot); 179 cout << endl; 180 } 181 182 template <typename T> 183 AVLTreeNode<T>* AVLTree<T>::search(AVLTreeNode<T>* tree, T key) const // 遞歸版搜索 184 { 185 if ( (tree == nullptr) || (tree->key == key) ) 186 return tree; 187 188 if ( key < tree->key ) 189 return search(tree->left, key); 190 else 191 return search(tree->right, key); 192 } 193 194 template <typename T> 195 AVLTreeNode<T>* AVLTree<T>::search(T key) const 196 { 197 return search(mRoot, key); 198 } 199 200 template <typename T> 201 AVLTreeNode<T>* AVLTree<T>::iterativeSearch(AVLTreeNode<T>* tree, T key) const // 非遞歸版搜索 202 { 203 while ( (tree != nullptr) && (tree->key != key) ) 204 { 205 if ( key < tree->key ) 206 tree = tree->left; 207 else 208 tree = tree->right; 209 } 210 211 return tree; 212 } 213 214 template <typename T> 215 AVLTreeNode<T>* AVLTree<T>::iterativeSearch(T key) const 216 { 217 return iterativeSearch(mRoot, key); 218 } 219 220 template <typename T> 221 AVLTreeNode<T>* AVLTree<T>::minimum(AVLTreeNode<T>* tree) const 222 { 223 if ( tree == nullptr ) 224 return nullptr; 225 226 while ( tree->left != nullptr ) 227 tree = tree->left; 228 229 return tree; 230 } 231 232 template <typename T> 233 T AVLTree<T>::minimum() const 234 { 235 AVLTreeNode<T> *ret = minimum(mRoot); 236 // if ( ret == nullptr ) 237 // THROW_EXCEPTION(EmptyTreeException, "The tree is empty ..."); 238 239 return ret->key; 240 } 241 242 template <typename T> 243 AVLTreeNode<T>* AVLTree<T>::maximum(AVLTreeNode<T>* tree) const 244 { 245 if ( tree == nullptr ) 246 return nullptr; 247 248 while ( tree->right != nullptr ) 249 tree = tree->right; 250 251 return tree; 252 } 253 254 template <typename T> 255 T AVLTree<T>::maximum() const 256 { 257 AVLTreeNode<T> *ret = maximum(mRoot); 258 // if ( ret == nullptr ) 259 // THROW_EXCEPTION(EmptyTreeException, "The tree is empty ..."); 260 261 return ret->key; 262 } 263 264 template <typename T> 265 AVLTreeNode<T>* AVLTree<T>::llRotation(AVLTreeNode<T>* tree) const // 左單旋,旋轉前左孩子有孩子右孩子沒有 266 { 267 AVLTreeNode<T>* ret; 268 269 ret = tree->left; 270 tree->left = tree->left->right; 271 ret->right = tree; 272 273 tree->height = max( height(tree->left), height(tree->right) ) + 1; 274 ret->height = max( height(ret->left), height(ret->right) ) + 1; 275 276 return ret; 277 } 278 279 template <typename T> 280 AVLTreeNode<T>* AVLTree<T>::rrRotation(AVLTreeNode<T>* tree) const // 右單旋,旋轉前右孩子有孩子左孩子沒有 281 { 282 AVLTreeNode<T>* ret; 283 284 ret = tree->right; 285 tree->right = tree->right->left; 286 ret->left = tree; 287 288 tree->height = max( height(tree->left), height(tree->right) ) + 1; 289 ret->height = max ( height(ret->left), height(ret->right) ) + 1; 290 291 return ret; 292 } 293 294 template <typename T> 295 AVLTreeNode<T>* AVLTree<T>::lrRotation(AVLTreeNode<T>* tree) const // 左右雙旋,先右旋,後左旋;tree左孩子的右孩子有後代,tree右孩子沒有後代 296 { 297 tree->left = rrRotation(tree->left); // 確保多出的後代在左子樹的左子樹上 298 299 return llRotation(tree); 300 } 301 302 template <typename T> 303 AVLTreeNode<T>* AVLTree<T>::rlRotation(AVLTreeNode<T>* tree) const // 右左雙旋,先左旋,後右旋;tree右孩子的左孩子有後代,tree左孩子沒有後代 304 { 305 tree->right = llRotation(tree->right); // 確保多出的後代在右子樹的右子樹上 306 307 return rrRotation(tree); 308 } 309 310 template <typename T> 311 AVLTreeNode<T>* AVLTree<T>::insert(AVLTreeNode<T>* &tree, T key) 312 { 313 if ( tree == nullptr ) // 建立新結點,下面的兩個else if會最終遞歸進入此if建立新結點 314 { 315 tree = new AVLTreeNode<T>(key, nullptr, nullptr); 316 ++mCount; 317 } 318 else if ( key < tree->key ) // 將新結點交給左子樹負責插入 319 { 320 tree->left = insert(tree->left, key); // 注意更新左結點,由於它可能旋轉或是新建立的結點 321 322 if ( height(tree->left) - height(tree->right) == 2) // 若是插入的左子樹超重 323 { 324 if ( key < tree->left->key ) 325 tree = llRotation(tree); // 若是插入的位置是左孩子的左孩子,左單旋 326 else 327 tree = lrRotation(tree); // 不然插入的位置是左孩子的右孩子,右單旋;最外層的else保證不會有重複值 328 } 329 } 330 else if ( key > tree->key ) // 將新結點交給右子樹負責插入 331 { 332 tree->right = insert(tree->right, key); 333 334 if ( height(tree->right) - height(tree->left) == 2 ) 335 { 336 if ( key > tree->right->key ) 337 tree = rrRotation(tree); 338 else 339 tree = rlRotation(tree); 340 } 341 } 342 else // 結點重複 343 ;//THROW_EXCEPTION(DuplicateDataException, "Can't create duplicate nodes ..."); 344 345 tree->height = max(height(tree->left), height(tree->right)) + 1; 346 347 return tree; 348 } 349 350 template <typename T> 351 void AVLTree<T>::insert(T key) 352 { 353 insert(mRoot, key); 354 } 355 356 template <typename T> 357 AVLTreeNode<T>* AVLTree<T>::remove(AVLTreeNode<T>* &tree, AVLTreeNode<T>* del) 358 { 359 if ( (tree == nullptr) || (del == nullptr) ) 360 return nullptr; 361 362 if ( del->key < tree->key ) // 交給左子樹刪除,最終進入最外層else 363 { 364 tree->left = remove(tree->left, del); // 更新可能旋轉的結點 365 366 if ( height(tree->right) - height(tree->left) == 2 ) // 刪除左子樹結點後若是右子樹超重 367 { 368 if ( height(tree->right->left) > height(tree->right->right) ) 369 tree = rlRotation(tree); // 右左重 370 else 371 tree = rrRotation(tree); // 右右重 372 } 373 } 374 else if ( del->key > tree->key ) // 交給右子樹刪除,最終進入最外層else 375 { 376 tree->right = remove(tree->right, del); 377 378 if ( height(tree->left) - height(tree->right) == 2 ) 379 { 380 if ( height(tree->left->right) - height(tree->left->left) == 2 ) 381 tree = lrRotation(tree); 382 else 383 tree = llRotation(tree); 384 } 385 } 386 else // 找到刪除節點 387 { 388 if ( (tree->left != nullptr) && (tree->right != nullptr) ) // 刪除點有兩個孩子,在較重的分支找替換者 389 { 390 if ( height(tree->left) > height(tree->right) ) // 左孩子重 391 { 392 AVLTreeNode<T>* lmax = maximum(tree->left); // 查找左邊最大者用於替換;若是用右邊的最小者,當右邊沒有孩子就會刪除自身致使不平衡 393 tree->key = lmax->key; // 採用值拷貝的方式刪除,不然你還要處理子結點 394 tree->left = remove(tree->left, lmax); // 更新可能旋轉的結點 395 } 396 else // 右孩子重 397 { 398 AVLTreeNode<T>* rmin = minimum(tree->right); 399 tree->key = rmin->key; 400 tree->right = remove(tree->right, rmin); 401 } 402 } 403 else // 刪除點孩子不足兩個,直接刪除並用孩子替換 404 { 405 AVLTreeNode<T>* tmp = tree; 406 tree = (tree->left != nullptr) ? tree->left : tree->right; // 替換刪除結點 407 delete tmp; 408 --mCount; 409 } 410 } 411 412 if ( tree != nullptr ) 413 tree->height = max(height(tree->left), height(tree->right)) + 1; 414 415 return tree; 416 } 417 418 template <typename T> 419 bool AVLTree<T>::remove(T key) 420 { 421 bool ret = false; 422 AVLTreeNode<T>* tmp; 423 424 if ( (tmp = search(mRoot, key)) != nullptr ) // 查找並刪除節點 425 { 426 mRoot = remove(mRoot, tmp); // remove()的返回值用於不能修改參數的編程語言,函數左邊的參數是左值mRoot的引用 427 ret = true; 428 } 429 430 return ret; 431 } 432 433 template <typename T> 434 void AVLTree<T>::printGraph(const void* Root, uint16_t m_keyStrLen) const 435 { 436 AVLTreeNode<T>const* node = static_cast<AVLTreeNode<T>const*>(Root); 437 if ( node == nullptr ) 438 return; 439 440 uint16_t height = node->height; // 要打印的樹總高度 441 uint16_t layer = 1; // 當前層,root爲第一層 442 uint64_t i, index; 443 uint64_t keyStrLen = m_keyStrLen; // i: 循環變量;index: 當前層最大結點數;keyStrLen: 結點輸出佔用字符寬度 444 queue<AVLTreeNode<T>const *> q; // 記錄每層的全部節點,包括nullptr 445 q.push(node); 446 cout << "輸出樹形關係圖!!!" << endl; 447 while ( true ) 448 { 449 cout << layer << "\t" << flush; // 輸出當前層號和當前層滿節點數 450 AVLTreeNode<T> const* tmp = nullptr; // 取出結點變量 451 index = 1ull<<(layer-1); 452 while ( index-- ) 453 { 454 tmp = q.front(); // 取出結點 455 q.pop(); 456 for ( i=0; i<((1ull<<(height-layer))-1ull)*keyStrLen; ++i ) // 結點前的填充 457 cout << " "; 458 cout << flush; 459 460 if ( tmp != nullptr ) // 打印有效結點 461 { 462 cout << right << setw(keyStrLen) << setfill('0') << tmp->key << flush; 463 464 if ( tmp->left != nullptr ) // 加入左結點 465 q.push(tmp->left); 466 else 467 q.push(nullptr); 468 469 if ( tmp->right != nullptr ) // 加入右節點 470 q.push(tmp->right); 471 else 472 q.push(nullptr); 473 } 474 else // 打印無效結點 475 { 476 for ( i=0; i<keyStrLen; ++i ) 477 cout << "#"; 478 cout << flush; 479 480 q.push(nullptr); // 若是結點是空的則爲其加入兩個空子結點 481 q.push(nullptr); 482 } 483 484 for ( i=0; i<((1ull<<(height-layer))-1ull)*keyStrLen; ++i ) // 結點後的填充 485 cout << " "; 486 cout << flush; 487 488 if ( q.size() != (1ull<<(layer)) ) 489 for ( i=0; i<keyStrLen; ++i ) // 兩節點間填充,由於父節點位於兩節點的中間上面,而不是其中一個的上面 490 cout << " "; 491 else 492 cout << "\t"; 493 cout << flush; 494 } 495 cout << layer << endl; // 輸出一層換行 496 497 if ( ++layer > height ) // while循環出口,當前層大於總高度時退出 498 break; 499 } 500 } 501 502 template <typename T> 503 void AVLTree<T>::printGraph(uint16_t keyStrLen) const 504 { 505 if ( mRoot == nullptr ) 506 return; 507 508 printGraph(mRoot, keyStrLen); 509 } 510 511 template <typename T> 512 void AVLTree<T>::printTree(AVLTreeNode<T> const* const tree, bool firstNode) const 513 { 514 if ( tree==nullptr ) 515 return; 516 517 bool static outTag[64] = {false}; // size = max layer limit; 518 uint8_t static layer = 0; 519 uint8_t i; 520 ++layer; 521 522 if ( layer >= 2 ) 523 { 524 for (i=2; i<layer; ++i ) 525 if ( outTag[i] ) 526 cout << "| "; 527 else 528 cout << " "; 529 if ( firstNode == true ) // 判斷左右結點,非二叉樹須要另外處理 530 cout << "L-------" << flush; 531 else 532 cout << "R-------" << flush; 533 } 534 cout << tree->key << endl; 535 536 for ( i=2-1; i>0; --i) // 從右往左輸出結點,即先打印最右邊結點,其次次右邊的結點;此循環不輸出最左邊的結點 537 { 538 if ( (tree->left+i) != nullptr ) // 注意樹的子結點指針必須是從左往右依次排列,中間不能有其它變量(left_1,left_2,left_3...left_n) 539 { // 若是你的子結點數量不定,必定要把後面的首個指針設爲nullptr 540 outTag[layer] = !firstNode; 541 printTree(tree->right, false); 542 } 543 } 544 if ( tree->left != nullptr ) // 輸出最左邊的結點 545 { 546 printTree(tree->left, true); 547 outTag[layer] = firstNode; 548 } 549 550 --layer; 551 } 552 553 template <typename T> 554 void AVLTree<T>::printTree() const 555 { 556 printTree(mRoot, true); // 右邊參數此時無心義 557 } 558 559 template <typename T> 560 void AVLTree<T>::destroy(AVLTreeNode<T>* &tree) const 561 { 562 if ( tree == nullptr ) 563 return; 564 565 if ( tree->left != nullptr ) 566 destroy(tree->left); 567 else 568 destroy(tree->right); 569 570 delete tree; 571 } 572 573 template <typename T> 574 void AVLTree<T>::destroy() 575 { 576 destroy(mRoot); 577 mRoot = nullptr; 578 mCount = 0; 579 } 580 581 template <typename T> 582 uint16_t AVLTree<T>::height(AVLTreeNode<T>* tree) const 583 { 584 if ( tree != nullptr ) 585 return tree->height; 586 587 return 0; 588 } 589 590 template <typename T> 591 uint16_t AVLTree<T>::height() const 592 { 593 return height(mRoot); 594 } 595 596 template <typename T> 597 uint64_t AVLTree<T>::getCount() const 598 { 599 return mCount; 600 } 601 602 template <typename T> 603 bool AVLTree<T>::rootIsNullptr() const 604 { 605 return mRoot == nullptr; 606 } 607 608 template <typename T> 609 T AVLTree<T>::getRootKey() const 610 { 611 // if ( mRoot == nullptr ) 612 // THROW_EXCEPTION(EmptyTreeException, "The tree is empty ..."); 613 614 return mRoot->key; 615 } 616 617 template <typename T> 618 AVLTree<T>::~AVLTree() 619 { 620 destroy(); 621 } 622 623 } 624 625 #endif // AVLTREE_H
main.cpppost
1 #include <iostream> 2 3 #include "AVLTree.h" 4 5 #include "Times.h" 6 7 using namespace std; 8 using namespace Viclib; 9 10 typedef uint64_t templateType; 11 typedef uint64_t sizeType; 12 13 int main(int argc, char* argv[]) 14 { 15 sizeType i, len = 5; 16 //uint16_t keyStrLen = 3; // 打印結點佔用的字符寬度,printGraph(node, keyStrLen) 17 18 if ( argc == 2 ) 19 len = static_cast<sizeType>(atoi(argv[1])); 20 else { 21 cout << "請輸入結點層數,注意內存大小" << endl; 22 cin >> len; 23 } 24 25 timingStart(); 26 27 templateType tmp = 0; 28 AVLTree<templateType>* tree = new AVLTree<templateType>(); 29 30 cout << "添加元素:\nkey\tcount\tlayer" << endl; 31 srand(static_cast<uint32_t>(time(nullptr))); 32 i = 0; 33 while ( tree->getCount() < (1ull<<len)-1 ) 34 { 35 do 36 { 37 tmp = static_cast<templateType>( 38 static_cast<sizeType>(rand()) 39 * static_cast<sizeType>(rand()) 40 * static_cast<sizeType>(rand()) 41 ); 42 } while(tree->iterativeSearch(tmp)); 43 tree->insert(tmp); 44 cout << tmp << "\t" << ++i << "\t" << tree->height() << endl; 45 if ( tree->height() >= static_cast<int>(len) ) // 限制樹的高度 46 break; 47 } 48 cout << endl; 49 50 cout << "先序遍歷:" << endl; 51 tree->preOrder(); 52 cout << endl; 53 cout << "中序遍歷:" << endl; 54 tree->inOrder(); 55 cout << endl; 56 cout << "後序遍歷:" << endl; 57 tree->postOrder(); 58 cout << endl; 59 cout << "廣度優先:" << endl; 60 tree->levelOrder(); 61 cout << endl; 62 63 // 輸出樹形描述的關係圖 64 // tree->printGraph(keyStrLen); 65 // cout << endl; 66 67 tree->printTree(); 68 cout << endl; 69 70 cout << "最小結點:" << tree->minimum() << endl; 71 cout << "最大結點:" << tree->maximum() << endl; 72 cout << "樹的高度:" << tree->height() << endl; 73 cout << "樹的結點數:" << tree->getCount() << endl; 74 75 cout << "\n開始刪除!!!\nkey\tlayer" << endl; 76 while ( !tree->rootIsNullptr() ) // 隨機數刪除 77 { 78 // tmp = static_cast<templateType>( 79 // static_cast<sizeType>(rand()) 80 // * static_cast<sizeType>(rand()) 81 // * static_cast<sizeType>(rand()) 82 // % ( (1ull<<len)-1) ); 83 if ( tree->remove(tree->getRootKey()) ) 84 cout << tmp << "\t" << tree->height() << "\t" << endl; 85 } 86 cout << endl; 87 88 cout << "刪除後輸出===" << endl; 89 cout << "樹的高度:" << tree->height() << endl; 90 cout << "樹的結點數:" << tree->getCount() << endl; 91 cout << "中序遍歷:" << endl; 92 tree->inOrder(); 93 94 cout << "\n銷燬AVL樹。" << endl; 95 tree->destroy(); 96 cout << "\n樹的高度:" << tree->height() << endl; 97 cout << "樹的結點數:" << tree->getCount() << endl; 98 delete tree; 99 tree = nullptr; 100 101 cout << endl; 102 timingEnd(); 103 cout << endl; 104 105 return 0; 106 }