完全二叉樹是效率很高的數據結構,完全二叉樹是由滿二叉樹而引出來的。對於深度為K的,有n個結點的二叉樹,若且唯若其每一個結點都與深度為K的滿二叉樹中編號從1至n的結點一一對應時稱之為完全二叉樹。
基本介紹
- 中文名:完全二叉樹
- 外文名:Complete Binary Tree
- 實質:效率很高的數據結構
- 特點:葉子結點只可能在最大的兩層出現
- 性質:度為1的點只有1個或0個
- 套用學科:計算機科學
定義,性質,特點,完全二叉樹判定,算法思路,代碼實現,
定義
若設二叉樹的深度為h,除第 h 層外,其它各層 (1~h-1) 的結點數都達到最大個數,第 h 層所有的結點都連續集中在最左邊,這就是完全二叉樹。
完全二叉樹是由滿二叉樹而引出來的。對於深度為K的,有n個結點的二叉樹,若且唯若其每一個結點都與深度為K的滿二叉樹中編號從1至n的結點一一對應時稱之為完全二叉樹。
(1)所有的葉結點都出現在第k層或k-l層(層次最大的兩層)
(2)對任一結點,如果其右子樹的最大層次為L,則其左子樹的最大層次為L或L+l。
一棵二叉樹至多只有最下面的兩層上的結點的度數可以小於2,並且最下層上的結點都集中在該層最左邊的若干位置上,則此二叉樹成為完全二叉樹,並且最下層上的結點都集中在該層最左邊的若干位置上,而在最後一層上,右邊的若干結點缺失的二叉樹,則此二叉樹成為完全二叉樹。
性質
如果一棵具有n個結點的深度為k的二叉樹,它的每一個結點都與深度為k的滿二叉樹中編號為1~n的結點一一對應,這棵二叉樹稱為完全二叉樹。
可以根據公式進行推導,假設n0是度為0的結點總數(即葉子結點數),n1是度為1的結點總數,n2是度為2的結點總數,則 :
①n= n0+n1+n2 (其中n為完全二叉樹的結點總數);又因為一個度為2的結點會有2個子結點,一個度為1的結點會有1個子結點,除根結點外其他結點都有父結點,
②n= 1+n1+2*n2 ;由①、②兩式把n2消去得:n= 2*n0+n1-1,由於完全二叉樹中度為1的結點數只有兩種可能0或1,由此得到n0=n/2 或 n0=(n+1)/2。
簡便來算,就是 n0=n/2,其中n為奇數時(n1=0)向上取整;n為偶數時(n1=1)向下取整。可根據完全二叉樹的結點總數計算出葉子結點數。
特點
葉子結點只可能在最大的兩層上出現,對任意結點,若其右分支下的子孫最大層次為L,則其左分支下的子孫的最大層次必為L 或 L+1;
var tree:array[1..n]of longint;{n:integer;n>=1}
對於tree[i],有如下特點:
(1)若i為奇數且i>1,那么tree的左兄弟為tree[i-1];
(2)若i為偶數且i<n,那么tree的右兄弟為tree[i+1];
(3)若i>1,tree的父親節點為tree[i div 2];
(4)若2*i<=n,那么tree的左孩子為tree[2*i];若2*i+1<=n,那么tree的右孩子為tree[2*i+1];
(5)若i>n div 2,那么tree[i]為葉子結點(對應於(3));
(6)若i<(n-1) div 2.那么tree[i]必有兩個孩子(對應於(4))。
(7)滿二叉樹一定是完全二叉樹,完全二叉樹不一定是滿二叉樹。
完全二叉樹第i層至多有2^(i-1)個節點,共i層的完全二叉樹最多有2^i-1個節點。
完全二叉樹的特點是:
1)只允許最後一層有空缺結點且空缺在右邊,即葉子結點只能在層次最大的兩層上出現;
2)對任一結點,如果其右子樹的深度為j,則其左子樹的深度必為j或j+1。 即度為1的點只有1個或0個
完全二叉樹判定
算法思路
判斷一棵樹是否是完全二叉樹的思路
1>如果樹為空,則直接返回錯
2>如果樹不為空:層序遍歷二叉樹
2.1>如果一個結點左右孩子都不為空,則pop該節點,將其左右孩子入佇列;
2.1>如果遇到一個結點,左孩子為空,右孩子不為空,則該樹一定不是完全二叉樹;
2.2>如果遇到一個結點,左孩子不為空,右孩子為空;或者左右孩子都為空;則該節點之後的佇列中的結點都為葉子節點;該樹才是完全二叉樹,否則就不是完全二叉樹;
2>如果樹不為空:層序遍歷二叉樹
2.1>如果一個結點左右孩子都不為空,則pop該節點,將其左右孩子入佇列;
2.1>如果遇到一個結點,左孩子為空,右孩子不為空,則該樹一定不是完全二叉樹;
2.2>如果遇到一個結點,左孩子不為空,右孩子為空;或者左右孩子都為空;則該節點之後的佇列中的結點都為葉子節點;該樹才是完全二叉樹,否則就不是完全二叉樹;
代碼實現
#include <iostream>#include <queue>using namespace std;template <class T>struct TreeNode{ T data; TreeNode<T> *left; TreeNode<T> *right; TreeNode(const T &x) : data(x), left(NULL), right(NULL) {}};template <class T>bool IsComplete(TreeNode<T> *root){ //1.樹為空,返回錯誤 if (root == NULL) { return false; } //2.樹不為空 queue<TreeNode<T> *> q; q.push(root); while (!q.empty()) { TreeNode<T> *top = q.front(); //2.1如果該節點兩個孩子都有,則直接pop if (top->left && top->right) { q.pop(); q.push(top->left); q.push(top->right); } //2.2如果該節點左孩子為空,右孩子不為空,則一定不是完全二叉樹 if (top->left == NULL && top->right) { return false; } //2.3如果該節點左孩子不為空,右孩子為空或者該節點為葉子節點,則該節點之後的所有結點都是葉子節點 if ((top->left && top->right == NULL) || (top->left == NULL && top->right == NULL)) { if (NULL != top->left && NULL == top->right) { q.push(top->left); } q.pop(); //則該節點之後的所有結點都是葉子節點 while (!q.empty()) { top = q.front(); if (top->left == NULL && top->right == NULL) { q.pop(); } else { return false; } } return true; } } return true;}//滿二叉樹void test1(){ // 1 // 2 3 // 4 5 6 7 TreeNode<int> *node1 = new TreeNode<int>(1); TreeNode<int> *node2 = new TreeNode<int>(2); TreeNode<int> *node3 = new TreeNode<int>(3); TreeNode<int> *node4 = new TreeNode<int>(4); TreeNode<int> *node5 = new TreeNode<int>(5); TreeNode<int> *node6 = new TreeNode<int>(6); TreeNode<int> *node7 = new TreeNode<int>(7); node1->left = node2; node1->right = node3; node2->left = node4; node2->right = node5; node3->left = node6; node3->right = node7; cout << IsComplete<int>(node1) << endl;}//二叉樹為空void test2(){ cout << IsComplete<int>(NULL) << endl;}//3.二叉樹不為空,也不是滿二叉樹,遇到一個結點左孩子為空,右孩子不為空void test3(){ // 1 // 2 3 // 4 5 7 TreeNode<int> *node1 = new TreeNode<int>(1); TreeNode<int> *node2 = new TreeNode<int>(2); TreeNode<int> *node3 = new TreeNode<int>(3); TreeNode<int> *node4 = new TreeNode<int>(4); TreeNode<int> *node5 = new TreeNode<int>(5); TreeNode<int> *node7 = new TreeNode<int>(7); node1->left = node2; node1->right = node3; node2->left = node4; node2->right = node5; node3->right = node7; cout << IsComplete<int>(node1) << endl;}//4.二叉樹不為空,也不是滿二叉樹,遇到葉子節點,則該葉子節點之後的所有結點都為葉子節點void test4(){ // 1 // 2 3 // 4 5 TreeNode<int> *node1 = new TreeNode<int>(1); TreeNode<int> *node2 = new TreeNode<int>(2); TreeNode<int> *node3 = new TreeNode<int>(3); TreeNode<int> *node4 = new TreeNode<int>(4); TreeNode<int> *node5 = new TreeNode<int>(5); node1->left = node2; node1->right = node3; node2->left = node4; node2->right = node5; cout << IsComplete<int>(node1) << endl;}//4.二叉樹不為空,也不是滿二叉樹,遇到左孩子不為空,右孩子為空的結點,則該節點之後的所有結點都為葉子節點void test5(){ // 1 // 2 3 // 4 5 6 TreeNode<int> *node1 = new TreeNode<int>(1); TreeNode<int> *node2 = new TreeNode<int>(2); TreeNode<int> *node3 = new TreeNode<int>(3); TreeNode<int> *node4 = new TreeNode<int>(4); TreeNode<int> *node5 = new TreeNode<int>(5); TreeNode<int> *node6 = new TreeNode<int>(6); node1->left = node2; node1->right = node3; node2->left = node4; node2->right = node5; node3->left = node6; cout << IsComplete<int>(node1) << endl;}int main(){ test1(); /*test2();*/ /*test3();*/ /*test4();*/ /*test5();*/ return 0;}