如何使用C語言實現(xiàn)平衡二叉樹數(shù)據(jù)結(jié)構(gòu)算法
前言
對于一個二叉排序樹而言
- 它們的結(jié)構(gòu)都是根據(jù)了二叉樹的特性從最左子樹開始在回到該結(jié)點上繼續(xù)往右結(jié)點走
- 通過該方式進行遞歸操作,并且該二叉排序樹的結(jié)構(gòu)也是從小到大依次顯示
- 那么我們假設(shè)a[10]={ 3,2,1,4,5,6,7,10,9,8 };我們需要查找改列表中的某一個結(jié)點的值
那么我們通過二叉排序樹的展示,會展示成如圖:
可以發(fā)現(xiàn),如果我們想通過二叉排序樹這個深度為8的樹來查找某個數(shù)
我們需要走到最后,這是最壞的打算
但是如果我們能將該樹改變?yōu)槠胶舛鏄?,如圖所示:
相比之下我們這個深度為4的平衡二叉樹在通過查詢值時是效率很平均且穩(wěn)定的,
那么通過上面的問題我們該怎么將這個二叉排序樹的變成平衡二叉樹是我們現(xiàn)在的問題了。
一、平衡二叉樹實現(xiàn)原理
對于平衡二叉樹的特性而言,我們在上面也有舉例到
- 在二叉排序樹中,按照它的從小到大的結(jié)構(gòu)所排序下來的結(jié)點,
- 當它的平衡因子的絕對值大于1的結(jié)點的根的子樹,我們稱為最小不平衡子樹。
而我們該如何去計算該二叉排序樹的平衡因子(平衡因子 = 左子樹 - 右子樹的值)呢?
我們可以結(jié)合上面那個最壞打算的二叉排序樹的生成方式進行逐一分析。
如圖:
可以看到當我們單步調(diào)試二叉排序樹的時候到值1的時候值3的平衡因子為2>1
- (左子樹有值2、值1所以是2-右子樹0所以等于2)
那么此時并不符合我們的平衡二叉樹的規(guī)則
那么我們需要通過某種形式去調(diào)換,使其符合我們的平衡二叉樹的規(guī)則
而圖2就是由圖1通過順時針而得到出來的,在這里我們稱為右旋
- 此時我們所有的平衡因子<2那么又恢復(fù)了正常
之后我們的值4繼續(xù)插入,因為值4>值3,所以根據(jù)二叉排序樹的特性是值3的右子樹
- 此時我們查看圖3發(fā)現(xiàn)所有的平衡因子<2,
那么也沒出現(xiàn)什么問題之后繼續(xù)插入下一個值5,因為值5>值4,所以值5會到值4的右子樹上。
如圖所示:
可以發(fā)現(xiàn)圖4中當插入值5的時候,根節(jié)點、值3的平衡因子的絕對值=2
- 所以是不符合平衡條件的所以此時我們通過逆時針的方式去旋轉(zhuǎn)值3下面的所有結(jié)點
- 在這里我們稱為左旋,得到的結(jié)果即為圖5所示了,且此時每個結(jié)點的平衡因子都<2,恢復(fù)平衡了
剩下就不多去演示了,整體是一個遞歸的過程
重要:在過程中反復(fù)的把不平衡的結(jié)點通過左、右旋的方式將其調(diào)整回平衡二叉樹。
要想深刻了解平衡二叉樹的實現(xiàn)原理請自行參考程杰著作的《大話數(shù)據(jù)結(jié)構(gòu)》。
二、平衡二叉樹實現(xiàn)算法
在類型上我們會通過二叉排序樹的基礎(chǔ)上添加一個bf,用于存儲平衡因子,而定義了一個Status則是判斷當前狀態(tài)的代碼。
代碼如下:
typedef int Status; /* Status是函數(shù)的類型,其值是函數(shù)結(jié)果狀態(tài)代碼,如OK等 */ /* 二叉樹的二叉鏈表結(jié)點結(jié)構(gòu)定義 */ typedef struct BiTNode /* 結(jié)點結(jié)構(gòu) */ { int data; /* 結(jié)點數(shù)據(jù) */ int bf; /* 結(jié)點的平衡因子 */ struct BiTNode* lchild, * rchild; /* 左右孩子指針 */ } BiTNode, * BiTree;
為了方便理解,我們前面的提到的左旋和右旋的方式,在c語言的代碼中該如何實現(xiàn)呢?下面我們就來將代碼展示出來。
代碼如下:
void R_Rotate(BiTree* P) { BiTree L; L = (*P)->lchild; /* L指向P的左子樹根結(jié)點 */ (*P)->lchild = L->rchild; /* L的右子樹掛接為P的左子樹 */ L->rchild = (*P); *P = L; /* P指向新的根結(jié)點 */ } /* 對以P為根的二叉排序樹作左旋處理, */ /* 處理之后P指向新的樹根結(jié)點,即旋轉(zhuǎn)處理之前的右子樹的根結(jié)點0 */ void L_Rotate(BiTree* P) { BiTree R; R = (*P)->rchild; /* R指向P的右子樹根結(jié)點 */ (*P)->rchild = R->lchild; /* R的左子樹掛接為P的右子樹 */ R->lchild = (*P); *P = R; /* P指向新的根結(jié)點 */ }
我們可以看到左旋和右旋的代碼很類似,所以我們通過了解了右旋的代碼的含義自然就可以反推出左旋的含義了。
此函數(shù)的意思是:
- 當某一個結(jié)點P處于一個不平衡狀態(tài)的時候,且平衡因子為正數(shù)需要右旋(順時針),并將它的左孩子結(jié)點定義為L。
- 將L的右子樹變成P的左子樹,再將P改成L的右子樹,最后將L替換P成功根結(jié)點。
如果文字還不了解我們可以將上述圖1、圖2的圖片進行推導(dǎo)。
如圖:
那么左旋的操作就是一個逆時針的方向轉(zhuǎn)換,代碼也是類似這里就不再演示了。
- 但是在這里我們還有另一種情況,就是當有兩個或以上的平衡因子的絕對值>=2的時候,它們的平衡因子的值可能為負也可能為正,那么這種情況我們就要考慮到雙旋操作了,
- 我們假設(shè)以樹根為頂點的左右子樹都需要考慮,且它們是相對的,所以我們可以通過了解左平衡旋轉(zhuǎn)處理也就繼而反推出右平衡旋轉(zhuǎn)處理了。
現(xiàn)在我們來看左平衡旋轉(zhuǎn)處理的代碼函數(shù):
#define LH +1 /* 左高 */ #define EH 0 /* 等高 */ #define RH -1 /* 右高 */ /* 對以指針T所指結(jié)點為根的二叉樹作右平衡旋轉(zhuǎn)處理, */ /* 本算法結(jié)束時,指針T指向新的根結(jié)點 */ void RightBalance(BiTree* T) { BiTree R, Rl; R = (*T)->rchild; /* R指向T的右子樹根結(jié)點 */ switch (R->bf) { /* 檢查T的右子樹的平衡度,并作相應(yīng)平衡處理 */ case RH: /* 新結(jié)點插入在T的右孩子的右子樹上,要作單左旋處理 */ (*T)->bf = R->bf = EH; L_Rotate(T); break; case LH: /* 新結(jié)點插入在T的右孩子的左子樹上,要作雙旋處理 */ Rl = R->lchild; /* Rl指向T的右孩子的左子樹根 */ switch (Rl->bf) { /* 修改T及其右孩子的平衡因子 */ case RH: (*T)->bf = LH; R->bf = EH; break; case EH: (*T)->bf = R->bf = EH; break; case LH: (*T)->bf = EH; R->bf = RH; break; } Rl->bf = EH; R_Rotate(&(*T)->rchild); /* 對T的右子樹作右旋平衡處理 */ L_Rotate(T); /* 對T作左旋平衡處理 */ } } /* 對以指針T所指結(jié)點為根的二叉樹作左平衡旋轉(zhuǎn)處理 */ /* 本算法結(jié)束時,指針T指向新的根結(jié)點 */ void LeftBalance(BiTree* T) { BiTree L, Lr; L = (*T)->lchild; /* L指向T的左子樹根結(jié)點 */ switch (L->bf) { /* 檢查T的左子樹的平衡度,并作相應(yīng)平衡處理 */ case LH: /* 新結(jié)點插入在T的左孩子的左子樹上,要作單右旋處理 */ (*T)->bf = L->bf = EH; R_Rotate(T); break; case RH: /* 新結(jié)點插入在T的左孩子的右子樹上,要作雙旋處理 */ Lr = L->rchild; /* Lr指向T的左孩子的右子樹根 */ switch (Lr->bf) { /* 修改T及其左孩子的平衡因子 */ 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); /* 對T的左子樹作左旋平衡處理 */ R_Rotate(T); /* 對T作右旋平衡處理 */ } }
如圖:
首先通過代碼可以看到,我們定義了三個常數(shù)變量,分別代碼1、0、-1。
然后呢因為是左平衡旋轉(zhuǎn)處理我們需要獲取當前圖片所示的P結(jié)點(也就是我之前假設(shè)的根節(jié)點的頂點)的左子樹,
這里要注意重點:
- 當我們這個函數(shù)可以執(zhí)行的時候已經(jīng)確認了當前子樹是不平衡的狀態(tài);
- 且結(jié)點N的值(即新插入結(jié)點) < 結(jié)點P的值(即上述圖片根結(jié)點P)和結(jié)點P->左子樹高于結(jié)點P->右子樹需要調(diào)整;
- 并且你還要記住此時進來的結(jié)點P才是不平衡狀態(tài),且通過該P結(jié)點->左子樹的平衡因子來進行分支判斷。
- 如果等于1了則執(zhí)行一次右旋操作,即上圖右旋推導(dǎo)的并且把它們的BF值(平衡因子)都改為0操作;
- 反之如果等于-1了則像上圖一樣進行雙旋操作;
- 不過在那之前還要通過獲取圖2上L的右子樹LR(貼合代碼上的Lr);
- 然后再判斷一下Lr的平衡因子的值并再次進行分支判斷;
- 修改根節(jié)點P(即代碼中的根節(jié)點T)根據(jù)Lr的BF值進行修改它們的BF值;
- 最后在將Lr的BF值賦為0,進行雙旋操作這里先左旋后右旋;
- 那么右平衡旋轉(zhuǎn)處理函數(shù)則取反。
三、全部代碼
#include "stdio.h" #include "stdlib.h" #include "io.h" #include "math.h" #include "time.h" #define OK 1 #define ERROR 0 #define TRUE 1 #define FALSE 0 #define MAXSIZE 100 /* 存儲空間初始分配量 */ typedef int Status; /* Status是函數(shù)的類型,其值是函數(shù)結(jié)果狀態(tài)代碼,如OK等 */ /* 二叉樹的二叉鏈表結(jié)點結(jié)構(gòu)定義 */ typedef struct BiTNode /* 結(jié)點結(jié)構(gòu) */ { int data; /* 結(jié)點數(shù)據(jù) */ int bf; /* 結(jié)點的平衡因子 */ struct BiTNode* lchild, * rchild; /* 左右孩子指針 */ } BiTNode, * BiTree; /* 對以p為根的二叉排序樹作右旋處理, */ /* 處理之后p指向新的樹根結(jié)點,即旋轉(zhuǎn)處理之前的左子樹的根結(jié)點 */ void R_Rotate(BiTree* P) { BiTree L; L = (*P)->lchild; /* L指向P的左子樹根結(jié)點 */ (*P)->lchild = L->rchild; /* L的右子樹掛接為P的左子樹 */ L->rchild = (*P); *P = L; /* P指向新的根結(jié)點 */ } /* 對以P為根的二叉排序樹作左旋處理, */ /* 處理之后P指向新的樹根結(jié)點,即旋轉(zhuǎn)處理之前的右子樹的根結(jié)點0 */ void L_Rotate(BiTree* P) { BiTree R; R = (*P)->rchild; /* R指向P的右子樹根結(jié)點 */ (*P)->rchild = R->lchild; /* R的左子樹掛接為P的右子樹 */ R->lchild = (*P); *P = R; /* P指向新的根結(jié)點 */ } #define LH +1 /* 左高 */ #define EH 0 /* 等高 */ #define RH -1 /* 右高 */ /* 對以指針T所指結(jié)點為根的二叉樹作左平衡旋轉(zhuǎn)處理 */ /* 本算法結(jié)束時,指針T指向新的根結(jié)點 */ void LeftBalance(BiTree* T) { BiTree L, Lr; L = (*T)->lchild; /* L指向T的左子樹根結(jié)點 */ switch (L->bf) { /* 檢查T的左子樹的平衡度,并作相應(yīng)平衡處理 */ case LH: /* 新結(jié)點插入在T的左孩子的左子樹上,要作單右旋處理 */ (*T)->bf = L->bf = EH; R_Rotate(T); break; case RH: /* 新結(jié)點插入在T的左孩子的右子樹上,要作雙旋處理 */ Lr = L->rchild; /* Lr指向T的左孩子的右子樹根 */ switch (Lr->bf) { /* 修改T及其左孩子的平衡因子 */ 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); /* 對T的左子樹作左旋平衡處理 */ R_Rotate(T); /* 對T作右旋平衡處理 */ } } /* 對以指針T所指結(jié)點為根的二叉樹作右平衡旋轉(zhuǎn)處理, */ /* 本算法結(jié)束時,指針T指向新的根結(jié)點 */ void RightBalance(BiTree* T) { BiTree R, Rl; R = (*T)->rchild; /* R指向T的右子樹根結(jié)點 */ switch (R->bf) { /* 檢查T的右子樹的平衡度,并作相應(yīng)平衡處理 */ case RH: /* 新結(jié)點插入在T的右孩子的右子樹上,要作單左旋處理 */ (*T)->bf = R->bf = EH; L_Rotate(T); break; case LH: /* 新結(jié)點插入在T的右孩子的左子樹上,要作雙旋處理 */ Rl = R->lchild; /* Rl指向T的右孩子的左子樹根 */ switch (Rl->bf) { /* 修改T及其右孩子的平衡因子 */ case RH: (*T)->bf = LH; R->bf = EH; break; case EH: (*T)->bf = R->bf = EH; break; case LH: (*T)->bf = EH; R->bf = RH; break; } Rl->bf = EH; R_Rotate(&(*T)->rchild); /* 對T的右子樹作右旋平衡處理 */ L_Rotate(T); /* 對T作左旋平衡處理 */ } } /* 若在平衡的二叉排序樹T中不存在和e有相同關(guān)鍵字的結(jié)點,則插入一個 */ /* 數(shù)據(jù)元素為e的新結(jié)點,并返回1,否則返回0。若因插入而使二叉排序樹 */ /* 失去平衡,則作平衡旋轉(zhuǎn)處理,布爾變量taller反映T長高與否。 */ Status InsertAVL(BiTree* T, int e, Status* taller) { if (!*T) { /* 插入新結(jié)點,樹“長高”,置taller為TRUE */ *T = (BiTree)malloc(sizeof(BiTNode)); (*T)->data = e; (*T)->lchild = (*T)->rchild = NULL; (*T)->bf = EH; *taller = TRUE; } else { if (e == (*T)->data) { /* 樹中已存在和e有相同關(guān)鍵字的結(jié)點則不再插入 */ *taller = FALSE; return FALSE; } if (e < (*T)->data) { /* 應(yīng)繼續(xù)在T的左子樹中進行搜索 */ if (!InsertAVL(&(*T)->lchild, e, taller)) /* 未插入 */ return FALSE; if (*taller) /* 已插入到T的左子樹中且左子樹“長高” */ switch ((*T)->bf) /* 檢查T的平衡度 */ { case LH: /* 原本左子樹比右子樹高,需要作左平衡處理 */ LeftBalance(T); *taller = FALSE; break; case EH: /* 原本左、右子樹等高,現(xiàn)因左子樹增高而使樹增高 */ (*T)->bf = LH; *taller = TRUE; break; case RH: /* 原本右子樹比左子樹高,現(xiàn)左、右子樹等高 */ (*T)->bf = EH; *taller = FALSE; break; } } else { /* 應(yīng)繼續(xù)在T的右子樹中進行搜索 */ if (!InsertAVL(&(*T)->rchild, e, taller)) /* 未插入 */ return FALSE; if (*taller) /* 已插入到T的右子樹且右子樹“長高” */ switch ((*T)->bf) /* 檢查T的平衡度 */ { case LH: /* 原本左子樹比右子樹高,現(xiàn)左、右子樹等高 */ (*T)->bf = EH; *taller = FALSE; break; case EH: /* 原本左、右子樹等高,現(xiàn)因右子樹增高而使樹增高 */ (*T)->bf = RH; *taller = TRUE; break; case RH: /* 原本右子樹比左子樹高,需要作右平衡處理 */ RightBalance(T); *taller = FALSE; break; } } } return TRUE; } int main(void) { int i; int a[10] = { 3,2,1,4,5,6,7,10,9,8 }; BiTree T = NULL; Status taller; for (i = 0; i < 10; i++) { InsertAVL(&T, a[i], &taller); } printf("本樣例建議斷點跟蹤查看平衡二叉樹結(jié)構(gòu)"); return 0; }
到此這篇關(guān)于如何使用C語言實現(xiàn)平衡二叉樹 的文章就介紹到這了,更多相關(guān)C語言實現(xiàn)平衡二叉樹內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++實現(xiàn)LeetCode(38.計數(shù)和讀法)
這篇文章主要介紹了C++實現(xiàn)LeetCode(38.計數(shù)和讀法),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-07-07C語言實現(xiàn)數(shù)組的循環(huán)左移,右移,翻轉(zhuǎn)的示例
今天小編就為大家分享一篇C語言實現(xiàn)數(shù)組的循環(huán)左移,右移,翻轉(zhuǎn)的示例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-07-07