欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語(yǔ)言之平衡二叉樹(shù)詳解

 更新時(shí)間:2023年04月25日 14:23:33   作者:Cukor丘克  
平衡二叉樹(shù)是具有平衡屬性的有序二叉樹(shù),本文主要介紹了C語(yǔ)言中的平衡二叉樹(shù),具有一定的參考價(jià)值,需要的小伙伴可以參考閱讀

什么是平衡二叉樹(shù)

平衡二叉樹(shù)是具有平衡屬性的有序二叉樹(shù),所謂的平衡即當(dāng)前樹(shù)的左右子樹(shù)高度差的絕對(duì)值不超過(guò)1。因?yàn)槠胶舛鏄?shù)是由蘇聯(lián)數(shù)學(xué)家Adelson-Velskii和Landis提出,所以又稱為AVL樹(shù)。

平衡二叉樹(shù)的基本特點(diǎn)

  • 是特殊的有序二叉樹(shù)
  • 左右子樹(shù)高度差的絕對(duì)值不超過(guò)1
  • 左右子樹(shù)仍然是平衡二叉樹(shù)

為什么會(huì)出現(xiàn)平衡二叉樹(shù)

在學(xué)習(xí)平衡二叉樹(shù)之前必定已經(jīng)學(xué)過(guò)有序二叉樹(shù),有序二叉樹(shù)的結(jié)構(gòu)特點(diǎn)就是將數(shù)據(jù)有序的排好,查找起來(lái)快,但是有序二叉樹(shù)有一個(gè)缺點(diǎn),就是當(dāng)節(jié)點(diǎn)呈現(xiàn)的狀態(tài)是一邊倒,那查找數(shù)據(jù)的時(shí)候就沒(méi)有發(fā)揮出二叉樹(shù)折半查找的優(yōu)勢(shì)了,這個(gè)時(shí)候是線性的查找(類似于鏈表的查找)。平衡二叉樹(shù)就是解決有序二叉樹(shù)一邊倒的問(wèn)題。如果有序二叉樹(shù)是平衡的,那么查找數(shù)據(jù)就很快。時(shí)間復(fù)雜度為O ( l o g n ) O(logn)O(logn)。這樣就充分發(fā)揮了二叉樹(shù)的優(yōu)勢(shì)。

二叉樹(shù)四種不平衡的情況

當(dāng)樹(shù)的左右子樹(shù)高度差的絕對(duì)值大于1的時(shí)候就需要進(jìn)行旋轉(zhuǎn)操作,將不平衡的樹(shù)變成平衡的樹(shù)。以下是會(huì)出現(xiàn)的四種不平衡的情況。

  • 左左不平衡
  • 右右不平衡
  • 左右不平衡
  • 右左不平衡

左左不平衡旋轉(zhuǎn)成平衡狀態(tài):

右右不平衡旋轉(zhuǎn)成平衡狀態(tài):

左右不平衡旋轉(zhuǎn)成平衡狀態(tài):

右左不平衡旋轉(zhuǎn)成平衡狀態(tài):

上面是圖解這四種不平衡狀態(tài)旋轉(zhuǎn)成平衡狀態(tài)的情況。

C語(yǔ)言實(shí)現(xiàn)平衡二叉樹(shù)

平衡二叉樹(shù)的結(jié)構(gòu)體描述:

#define Ty int  //以整型數(shù)據(jù)為例
typedef struct Node
{
	Ty data;             //數(shù)據(jù)
	int height;          //高度
	struct Node* LChild; //左子樹(shù)
	struct Node* RChild; //右子樹(shù)
}Node,AVLTree;

初始化函數(shù):

AVLTree* creatAVLTree(Ty data)
{
    AVLTree* tree = (AVLTree*)malloc(sizeof(AVLTree));
    assert(tree);
    tree->data = data;
    tree->height = 0;
    tree->LChild = NULL;
    tree->RChild = NULL;
    return tree;
}

輔助宏函數(shù):

//輔助函數(shù)
#define HEIGHT(x) ((x==NULL)?(-1):(x->height))
#define MAX(a,b)  ((a>b)?(a):(b))
//獲取樹(shù)的新高度
#define GET_NEW_HEIGHT(x) (MAX(HEIGHT(x->LChild),HEIGHT(x->RChild)) + 1)

使用宏函數(shù)的好處是運(yùn)行過(guò)程中不需要進(jìn)行函數(shù)壓棧的操作,效率快一點(diǎn)。

前序遍歷平衡二叉樹(shù):

//前序打印
void show_pre(AVLTree* root)
{
	if(root==NULL)
		return;
	printf("data:%d\theight:%d\n",root->data,root->height);
	show_pre(root->LChild);
	show_pre(root->RChild);
}

使用前序遍歷能更好地看出節(jié)點(diǎn)的高度,更方便還原平衡二叉樹(shù)。

四種不平衡狀態(tài)旋轉(zhuǎn)的算法實(shí)現(xiàn):

算法核心思想:找到新根的位置,然后進(jìn)行對(duì)應(yīng)的調(diào)整,最后返回新根的地址,這樣就實(shí)現(xiàn)了旋轉(zhuǎn)的操作,因?yàn)樾D(zhuǎn)后節(jié)點(diǎn)的高度改變了,所以在返回之前先調(diào)整一下節(jié)點(diǎn)的高度。

例如:左左不平衡進(jìn)行旋轉(zhuǎn)操作

因?yàn)槭亲笞蟛黄胶猓孕赂奈恢檬钱?dāng)前根的左子樹(shù),那就使用一個(gè)指針(newRoot)去接收當(dāng)前根的左子樹(shù),然后使勁將當(dāng)前根拉下來(lái),讓新根代替當(dāng)前根的位置,那就必須將當(dāng)前根的LChild指向newRoot的右子樹(shù)(因?yàn)閚ewRoot不一定是空的所以不能直接讓curRoot→LChild指向空)。最后就是將newRoot→RChild指向curRoot這樣就把位置調(diào)整好了。在返回newRoot之前把curRoot和newRoot的高度調(diào)整一下。保持樹(shù)的準(zhǔn)確性。

其他的不平衡情況也是類似的操作。

//出現(xiàn)不平衡的情況
//左左不平衡
Node *LL_Rotation(Node *curRoot)
{
	Node *newRoot = curRoot->LChild;
	curRoot->LChild = newRoot->RChild;
	newRoot->RChild = curRoot;

	curRoot->height = GET_NEW_HEIGHT(curRoot);
	newRoot->height = GET_NEW_HEIGHT(newRoot);
	return newRoot;
}

//右右不平衡
Node *RR_Rotation(Node *curRoot)
{
	Node *newRoot = curRoot->RChild;
	curRoot->RChild = newRoot->LChild;
	newRoot->LChild = curRoot;

	curRoot->height = GET_NEW_HEIGHT(curRoot);
	newRoot->height = GET_NEW_HEIGHT(newRoot);
	return newRoot;
}
//左右不平衡
Node *LR_Rotation(Node *curRoot)
{
	curRoot->LChild = RR_Rotation(curRoot->LChild);
	return LL_Rotation(curRoot);
}
//右左不平衡
Node *RL_Rotation(Node *curRoot)
{
	curRoot->RChild = LL_Rotation(curRoot->RChild);
	return RR_Rotation(curRoot);
}

平衡二叉樹(shù)的插入操作:

插入操作需要考慮到四種情況:

  • 當(dāng)前節(jié)點(diǎn)是空的
  • 要插入進(jìn)來(lái)的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)小
  • 要插入進(jìn)來(lái)的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)大
  • 要插入進(jìn)來(lái)的數(shù)據(jù)和當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)一樣大

情況一的解決方案:直接申請(qǐng)一個(gè)節(jié)點(diǎn)內(nèi)存。

情況二的解決方案:遞歸往左邊跑,然后跑到對(duì)應(yīng)的位置就申請(qǐng)內(nèi)存,插入完成后判斷需不需要調(diào)整。

情況三的解決方案:遞歸往右邊跑,然后跑到對(duì)應(yīng)的位置就申請(qǐng)內(nèi)存,插入完成后判斷需不需要調(diào)整。

情況四的解決方案:因?yàn)槲覀冏龅氖且豢脹](méi)有重復(fù)數(shù)據(jù)的平衡二叉樹(shù)所以遇到這種情況的時(shí)候不進(jìn)行插入操作。當(dāng)然如果做的是一棵可以有重復(fù)數(shù)據(jù)的平衡二叉樹(shù),那遇到這種情況的時(shí)候可以個(gè)人的想法放左邊放右邊都可以。

源代碼:

//插入數(shù)據(jù)
Node *insertNode(Node *curRoot, Ty data)
{
	//插入分有四個(gè)大情況
	if (curRoot == NULL)			  //當(dāng)前節(jié)點(diǎn)是空的
		curRoot = creatAVLTree(data); //如果是空就直接創(chuàng)建一個(gè)新的節(jié)點(diǎn)
	else if (data < curRoot->data)	  //要插入的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)小
	{
		//往左邊跑
		curRoot->LChild = insertNode(curRoot->LChild, data);
		//插入完成之后,判斷需不需要調(diào)整樹(shù)
		if (HEIGHT(curRoot->LChild) - HEIGHT(curRoot->RChild) == 2)
			//因?yàn)椴迦氲奈恢迷谧筮?,所以插入完成之后,左子?shù)的高度大于等于右子樹(shù)高度
			curRoot = data < curRoot->LChild->data ? LL_Rotation(curRoot) : LR_Rotation(curRoot);
	}
	else if (data > curRoot->data) //要插入的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)大
	{
		//往右邊跑
		curRoot->RChild = insertNode(curRoot->RChild, data);
		if (HEIGHT(curRoot->RChild) - HEIGHT(curRoot->LChild) == 2)
			//因?yàn)椴迦氲奈恢迷谟疫?,所以插入完成之后,右子?shù)的高度大于等于左子樹(shù)高度
			curRoot = data > curRoot->RChild->data ? RR_Rotation(curRoot) : RL_Rotation(curRoot);
	}
	else //要插入的數(shù)據(jù)和當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)一樣大
		printf("無(wú)法插入數(shù)據(jù)\n");
	//獲取新高度
	curRoot->height = GET_NEW_HEIGHT(curRoot);
	return curRoot; //插入完成之后返回當(dāng)前節(jié)點(diǎn)的指針
}

平衡二叉樹(shù)的刪除操作:

刪除操作也是要考慮四個(gè)大情況:

  • 當(dāng)前節(jié)點(diǎn)是空的
  • 要?jiǎng)h除的數(shù)據(jù)比當(dāng)前數(shù)據(jù)小
  • 要?jiǎng)h除的數(shù)據(jù)比當(dāng)前數(shù)據(jù)大
  • 要?jiǎng)h除的數(shù)據(jù)和當(dāng)前數(shù)據(jù)一樣大

情況一的解決方案:沒(méi)有刪除的必要了,結(jié)束掉函數(shù)

情況二的解決方案:往左邊遞歸找到對(duì)應(yīng)位置,然后進(jìn)行刪除操作

情況三的解決方案:往右邊遞歸找到對(duì)應(yīng)位置,然后進(jìn)行刪除操作

情況四的解決方案:針對(duì)這個(gè)情況又要分為兩個(gè)小情況

  • 當(dāng)前節(jié)點(diǎn)的左子樹(shù)和右子樹(shù)都存在
  • 當(dāng)前節(jié)點(diǎn)的左右子樹(shù)至多有一個(gè)存在

具體解決方案看代碼和注釋

源代碼:

//查找節(jié)點(diǎn)
//找最大節(jié)點(diǎn)
Node *maxNode(Node *curRoot)
{
	if (curRoot == NULL)
		return NULL;
	//往右邊找
	while (curRoot->RChild)
		curRoot = curRoot->RChild;
	return curRoot;
}

//找最小節(jié)點(diǎn)
Node *minNode(Node *curRoot)
{
	if (curRoot == NULL)
		return NULL;
	while (curRoot->LChild)
		curRoot = curRoot->LChild;
	return curRoot;
}

//刪除數(shù)據(jù)
Node *deleteNode(Node *curRoot, Ty data)
{
	//刪除數(shù)據(jù)有四個(gè)大情況
	if (curRoot == NULL)	  //當(dāng)前節(jié)點(diǎn)是空的
		return NULL;		  //刪除了個(gè)寂寞直接結(jié)束掉整個(gè)函數(shù)
	if (data < curRoot->data) //要?jiǎng)h除的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)小
	{
		//往左邊跑
		curRoot->LChild = deleteNode(curRoot->LChild, data);
		//獲取新高度
		curRoot->height = GET_NEW_HEIGHT(curRoot);
		//判斷需不需要調(diào)整
		if (HEIGHT(curRoot->RChild) - HEIGHT(curRoot->LChild) == 2)
			curRoot = HEIGHT(curRoot->RChild->LChild) > HEIGHT(curRoot->RChild->RChild) ? RL_Rotation(curRoot) : RR_Rotation(curRoot);
	}
	else if (data > curRoot->data) //要?jiǎng)h除的數(shù)據(jù)比當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)大
	{
		//往右邊跑
		curRoot->RChild = deleteNode(curRoot->RChild, data);
		curRoot->height = GET_NEW_HEIGHT(curRoot);
		if (HEIGHT(curRoot->LChild) - HEIGHT(curRoot->RChild) == 2)
			curRoot = HEIGHT(curRoot->LChild->RChild) > HEIGHT(curRoot->LChild->LChild) ? LR_Rotation(curRoot) : LL_Rotation(curRoot);
	}
	else
	{ //要?jiǎng)h除的數(shù)據(jù)和當(dāng)前節(jié)點(diǎn)的數(shù)據(jù)一樣大
		//針對(duì)于curRoot這個(gè)節(jié)點(diǎn)做刪除操作
		//主要有兩個(gè)主要的情況
		if (curRoot->LChild && curRoot->RChild) // curRoot有左子樹(shù)和右子樹(shù)
		{
			//先判斷左右子樹(shù)的高度,將高度比較高的子樹(shù)的節(jié)點(diǎn)拿到上面來(lái)
			if (HEIGHT(curRoot->LChild) > HEIGHT(curRoot->RChild))
			{ //左子樹(shù)的高度比右子樹(shù)的高度高
				//找到左子樹(shù)的最大節(jié)點(diǎn)
				Node *max = maxNode(curRoot->LChild);
				//找到之后就將max的數(shù)據(jù)替換curRoot的數(shù)據(jù)
				curRoot->data = max->data;
				//賦值完成之后繼續(xù)遞歸,然后釋放掉max對(duì)應(yīng)的節(jié)點(diǎn),不能直接在這里釋放,因?yàn)橐{(diào)整樹(shù)的高度
				curRoot->LChild = deleteNode(curRoot->LChild, max->data);
			}
			else
			{ //左子樹(shù)的高度小于等于右子樹(shù)的高度
				//找到右子樹(shù)的最小節(jié)點(diǎn)
				Node *min = minNode(curRoot->RChild);
				curRoot->data = min->data;
				curRoot->RChild = deleteNode(curRoot->RChild, min->data);
			}
		}
		else //上一種情況的否定,即curRoot沒(méi)有子樹(shù)或者只有一邊
		{
			//釋放內(nèi)存
			Node *temp = curRoot;
			// curRoot拿到存在的子樹(shù)
			curRoot = curRoot->LChild ? curRoot->LChild : curRoot->RChild;
			free(temp);
		}
	}

	return curRoot; //刪除完成之后就返回當(dāng)前節(jié)點(diǎn)
}

主函數(shù)測(cè)試:

int main()
{
	AVLTree *tree = NULL;
	for (int i = 1; i < 10; i++)
		tree = insertNode(tree, i);
	show_pre(tree); //前序打印樹(shù)
	printf("----------------------------\n");
	//刪除6這個(gè)節(jié)點(diǎn)
	tree = deleteNode(tree, 6);
	show_pre(tree);
	printf("程序結(jié)束\n");
	return 0;
}

運(yùn)行結(jié)果:

刪除前和刪除后的平衡二叉樹(shù):

到此這篇關(guān)于C語(yǔ)言之平衡二叉樹(shù)詳解的文章就介紹到這了,更多相關(guān)C++平衡二叉樹(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C/C++實(shí)現(xiàn)磁盤相關(guān)操作的示例代碼

    C/C++實(shí)現(xiàn)磁盤相關(guān)操作的示例代碼

    這篇文章主要為大家詳細(xì)介紹了C/C++如何實(shí)現(xiàn)磁盤相關(guān)操作,例如遍歷磁盤容量、實(shí)現(xiàn)磁盤格式化、移除指定磁盤等,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-11-11
  • 一文搞懂C++多態(tài)的用法

    一文搞懂C++多態(tài)的用法

    C++多態(tài)是在繼承的基礎(chǔ)上實(shí)現(xiàn)的,了解多態(tài)之前我們需要掌握一定的C++繼承的知識(shí),本文將介紹C++中多態(tài)的概念,構(gòu)成條件以及用法,感興趣的可以學(xué)習(xí)一下
    2022-04-04
  • C++內(nèi)存對(duì)象布局小測(cè)試

    C++內(nèi)存對(duì)象布局小測(cè)試

    這篇文章主要介紹了C++內(nèi)存對(duì)象布局小測(cè)試,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12
  • C語(yǔ)言動(dòng)態(tài)規(guī)劃點(diǎn)殺dp算法LeetCode炒股習(xí)題案例解析

    C語(yǔ)言動(dòng)態(tài)規(guī)劃點(diǎn)殺dp算法LeetCode炒股習(xí)題案例解析

    這篇文章主要介紹為了C語(yǔ)言動(dòng)態(tài)規(guī)劃點(diǎn)殺dp算法,本文以LeetCode炒股習(xí)題案例來(lái)為大家進(jìn)行詳細(xì)解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2022-02-02
  • C++實(shí)現(xiàn)雙向起泡排序算法

    C++實(shí)現(xiàn)雙向起泡排序算法

    這篇文章主要為大家詳細(xì)介紹了如何利用C++實(shí)現(xiàn)雙向起泡排序算法,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以嘗試一下
    2022-11-11
  • C語(yǔ)言之字符串模糊查詢方法的實(shí)現(xiàn)

    C語(yǔ)言之字符串模糊查詢方法的實(shí)現(xiàn)

    本篇文章主要為大家介紹字符串模糊查詢的C語(yǔ)言程序編寫方法,有需要的朋友可以參考下
    2015-07-07
  • Qt 鼠標(biāo)/觸屏繪制平滑曲線(支持矢量/非矢量方式)

    Qt 鼠標(biāo)/觸屏繪制平滑曲線(支持矢量/非矢量方式)

    這篇文章主要介紹了Qt 鼠標(biāo)/觸屏繪制平滑曲線(支持矢量/非矢量方式),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • c語(yǔ)言實(shí)現(xiàn)兩個(gè)值互相交換的函數(shù)

    c語(yǔ)言實(shí)現(xiàn)兩個(gè)值互相交換的函數(shù)

    本文通過(guò)代碼給大家介紹c語(yǔ)言實(shí)現(xiàn)兩個(gè)值互相交換的函數(shù),通過(guò)實(shí)例代碼給大家講解的很詳細(xì),具有一定的參考借鑒價(jià)值,對(duì)c語(yǔ)言兩個(gè)值互換函數(shù)相關(guān)知識(shí)感興趣的朋友一起看看吧
    2021-05-05
  • C++回溯算法中組合的相關(guān)問(wèn)題分析

    C++回溯算法中組合的相關(guān)問(wèn)題分析

    回溯算法并不是什么高效的算法,因?yàn)楸举|(zhì)上時(shí)去遍歷所有元素,找出所有可能,然后選出需要的答案。那為什么還要回溯法,簡(jiǎn)單來(lái)說(shuō),不是所有的問(wèn)題都能用什么巧妙的方法來(lái)解決的
    2023-03-03
  • C/C++實(shí)現(xiàn)圖形學(xué)掃描線填充算法

    C/C++實(shí)現(xiàn)圖形學(xué)掃描線填充算法

    這篇文章主要介紹了C/C++實(shí)現(xiàn)圖形學(xué)掃描線填充算法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-04-04

最新評(píng)論