平衡二叉樹(shù)的實(shí)現(xiàn)實(shí)例
/*
首先平衡二叉樹(shù)是一個(gè)二叉排序樹(shù);
其基本思想是:
在構(gòu)建二叉排序樹(shù)的過(guò)程中,當(dāng)每插入一個(gè)節(jié)點(diǎn)時(shí),
先檢查是否因?yàn)椴迦攵茐牧藰?shù)的平衡性,若是,
找出最小不平衡樹(shù),進(jìn)行適應(yīng)的旋轉(zhuǎn),使之成為新的平衡二叉樹(shù)。
*/
#include<cstdio>
#include<cstdlib>
#define LH 1
#define EH 0
#define RH -1
using namespace std;
typedef struct BTNode
{
int data;
int BF;//平衡因子(balance factor)
struct BTNode *lchild,*rchild;
}BTNode,*BTree;
void R_Rotate(BTree *p)//以p為根節(jié)點(diǎn)的二叉排序樹(shù)進(jìn)行右旋轉(zhuǎn)
{
BTree L;
L=(*p)->lchild;
(*p)->lchild=L->rchild;
L->rchild=(*p);
*p=L;//p指向新的根節(jié)點(diǎn)
}
void L_Rotate(BTree *p)//以p為根節(jié)點(diǎn)的二叉排序樹(shù)進(jìn)行左旋轉(zhuǎn)
{
BTree R;
R=(*p)->rchild;
(*p)->rchild=R->lchild;
R->lchild=(*p);
*p=R;
}
void LeftBalance(BTree *T)
{
BTree L,Lr;
L=(*T)->lchild;
switch(L->BF)
{
//檢查T的左子樹(shù)平衡度,并作相應(yīng)的平衡處理
case LH://新節(jié)點(diǎn)插入在T的左孩子的左子樹(shù)上,做單右旋處理
(*T)->BF=L->BF=EH;
R_Rotate(T);
break;
case RH://新插入節(jié)點(diǎn)在T的左孩子的右子樹(shù)上,做雙旋處理
Lr=L->rchild;
switch(Lr->BF)
{
case LH:
(*T)->BF=RH;
L->BF=EH;
break;
case EH:
(*T)->BF=L->BF=EH;
break;
case RH:
(*T)->BF=EH;
L->BF=LH;
break;
}
Lr->BF=EH;
L_Rotate(&(*T)->lchild);
R_Rotate(T);
}
}
void RightBalance(BTree *T)
{
BTree R,Rl;
R=(*T)->rchild;
switch(R->BF)
{
case RH://新節(jié)點(diǎn)插在T的右孩子的右子樹(shù)上,要做單左旋處理
(*T)->BF=R->BF=EH;
L_Rotate(T);
break;
case LH://新節(jié)點(diǎn)插在T的右孩子的左子樹(shù)上,要做雙旋處理
Rl=R->lchild;
switch(Rl->BF)
{
case LH:
(*T)->BF=EH;
R->BF=RH;
break;
case EH:
(*T)->BF=R->BF=EH;
break;
case RH:
(*T)->BF=LH;
R->BF=EH;
break;
}
Rl->BF=EH;
R_Rotate(&(*T)->rchild);
L_Rotate(T);
}
}
bool InsertAVL(BTree *T,int e,bool *taller)//變量taller反應(yīng)T長(zhǎng)高與否
{
if(!*T)
{
*T=(BTree)malloc(sizeof(BTNode));
(*T)->data=e;
(*T)->lchild=(*T)->rchild=NULL;
(*T)->BF=EH;
*taller=true;
}
else
{
if(e==(*T)->data)//不插入
{
*taller=false;
return false;
}
if(e<(*T)->data)
{
if(!InsertAVL(&(*T)->lchild,e,taller))//未插入
return false;
if(*taller)//以插入左子樹(shù),且左子樹(shù)變高
{
switch((*T)->BF)
{
case LH://原本左子樹(shù)比右子樹(shù)高,需要做左平衡處理
LeftBalance(T);
*taller=false;
break;
case EH://原本左右子樹(shù)等高,現(xiàn)因左子樹(shù)增高而樹(shù)增高
(*T)->BF=LH;
*taller=true;
break;
case RH://原本右子樹(shù)比左子樹(shù)高,現(xiàn)在左右子樹(shù)等高
(*T)->BF=EH;
*taller=false;
break;
}
}
}
else
{
//應(yīng)在T的右子樹(shù)中搜尋
if(!InsertAVL(&(*T)->rchild,e,taller))
return false;
if(*taller)//插入右子樹(shù),且右子樹(shù)長(zhǎng)高
{
switch((*T)->BF)
{
case LH://原本左子樹(shù)比右子樹(shù)高,現(xiàn)在左右子樹(shù)等高
(*T)->BF=EH;
*taller=false;
break;
case EH://原本左右子樹(shù)等高,現(xiàn)在右子樹(shù)變高
(*T)->BF=RH;
*taller=true;
break;
case RH://原本右子樹(shù)比左子樹(shù)高,現(xiàn)在需做右平衡處理
RightBalance(T);
*taller=false;
break;
}
}
}
}
return true;
}
bool Find(BTree T,int key)
{
if(!T)
return false;
else if(T->data==key)
return true;
else if(T->data<key)
return Find(T->rchild,key);
else
return Find(T->lchild,key);
}
void Output(BTree T)
{
if(T)
{
printf("%d",T->data);
if(T->lchild||T->rchild)
{
printf("(");
Output(T->lchild);
printf(",");
Output(T->rchild);
printf(")");
}
}
}
int main(int argc,char *argv[])
{
int i;
int A[]={3,2,1,4,5,6,7,10,9,8};
BTree T=NULL;
bool taller;
for(i=0;i<sizeof(A)/sizeof(int);i++)
InsertAVL(&T,A[i],&taller);
Output(T);
printf("\n");
if(Find(T,6))
printf("6 is find in the AVL tree!\n");
else
printf("6 is not find in the AVL tree!\n");
return 0;
}
上一篇:wince程序防止創(chuàng)建多個(gè)實(shí)例實(shí)現(xiàn)互斥作用
欄 目:C語(yǔ)言
本文標(biāo)題:平衡二叉樹(shù)的實(shí)現(xiàn)實(shí)例
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/3798.html
您可能感興趣的文章
- 01-10深入二叉樹(shù)兩個(gè)結(jié)點(diǎn)的最低共同父結(jié)點(diǎn)的詳解
- 01-10深入理解二叉樹(shù)的非遞歸遍歷
- 01-10深入遍歷二叉樹(shù)的各種操作詳解(非遞歸遍歷)
- 01-10判斷整數(shù)序列是否為二元查找樹(shù)的后序遍歷結(jié)果的解決方法
- 01-10探討:C++實(shí)現(xiàn)鏈?zhǔn)蕉鏄?shù)(用非遞歸方式先序,中序,后序遍歷二叉樹(shù)
- 01-10如何在二叉樹(shù)中找出和為某一值的所有路徑
- 01-10深入linux下遍歷目錄樹(shù)的方法總結(jié)分析
- 01-10先序遍歷二叉樹(shù)的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)深入解析
- 01-10二叉搜索樹(shù)的插入與刪除(詳細(xì)解析)
- 01-10二叉查找樹(shù)的插入,刪除,查找


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫(xiě)分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫(xiě)函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?