二叉樹之AVL樹

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 }
相關文章
相關標籤/搜索