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

C++實(shí)現(xiàn)二叉樹及堆的示例代碼

 更新時(shí)間:2021年04月20日 15:49:04   作者:一枚快樂(lè)的野指針  
這篇文章主要介紹了C++實(shí)現(xiàn)二叉樹及堆的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

1 樹

樹是一種非線性數(shù)據(jù)結(jié)構(gòu),它是由n個(gè)有限結(jié)點(diǎn)組成的具有層次關(guān)系的集合。把它叫樹是因?yàn)樗歉?,葉子朝下的
來(lái)上圖瞧瞧

在這里插入圖片描述

1.1 樹的相關(guān)名詞

在這里插入圖片描述

2 二叉樹

2.1 二叉樹的概念

一顆二叉樹是結(jié)點(diǎn)的一個(gè)有限集合,該集合或者為空,或者是由一個(gè)根結(jié)點(diǎn)加上兩棵分別稱為左子樹和右子樹。
如圖所示:

在這里插入圖片描述

二叉樹有以下特點(diǎn):

1、每個(gè)二叉樹最多有兩顆子樹,所以二叉樹不存在度為2的結(jié)點(diǎn)。
2、二叉樹的子樹有左右之分,其子樹的順序不能顛倒。

2.2 二叉樹的性質(zhì)

1、若規(guī)定根結(jié)點(diǎn)的層數(shù)為第一層,則一顆非空二叉樹的第i層上最多有z^(k-1)個(gè)結(jié)點(diǎn)
2、若規(guī)定根結(jié)點(diǎn)的層數(shù)為第一層,則深度為h的二叉樹的最大結(jié)點(diǎn)數(shù)是2^k-1個(gè)。
3、對(duì)于任何一顆二叉樹,度為0的結(jié)點(diǎn)(葉子結(jié)點(diǎn))的個(gè)數(shù)為n0 ,度為2的結(jié)點(diǎn)個(gè)數(shù)為n2則一定有,n0 = n2 + 1。
4、若規(guī)定根結(jié)點(diǎn)的層數(shù)為第一層,具有N個(gè)結(jié)點(diǎn)的滿二叉樹的深度h = log2(N+1)[說(shuō)明:以2為底的N+1的對(duì)數(shù)],這個(gè)可以由性質(zhì)2推導(dǎo)得到。
5、對(duì)于具有n個(gè)結(jié)點(diǎn)的完全二叉樹,如果按照從上到下從左到右的數(shù)組順序?qū)λ薪Y(jié)點(diǎn)從0開始編號(hào),即對(duì)于數(shù)組下標(biāo)i的結(jié)點(diǎn)有:
1)i>1,i位置的父結(jié)點(diǎn)的在數(shù)組中的下標(biāo)為(i-1)/2.
2)i位置結(jié)點(diǎn)的左孩子結(jié)點(diǎn)的下標(biāo)為2i+1,右結(jié)點(diǎn)下標(biāo)為2i+2。

2.3 特殊的二叉樹

1、滿二叉樹:一個(gè)二叉樹,如果每一層的結(jié)點(diǎn)數(shù)都達(dá)到了最大,如果這個(gè)二叉樹的層次是k,則其結(jié)點(diǎn)數(shù)是2^k-1。
2、完全二叉樹:用最直白的話來(lái)說(shuō)就是,樹的深度或者高度為k,前k-1層的結(jié)點(diǎn)都是滿的,只有最后的第k層不滿但是從左到右的結(jié)點(diǎn)必須是連續(xù)的。
其實(shí)滿二叉樹是一種特殊的完全二叉樹。
如圖所示:

在這里插入圖片描述

圖為完全二叉樹,要是最后一層全滿則為滿二叉樹。
滿足:
2^k-1-x = N;
x的取值范圍是[0,2^(k-1)-1]

3 堆

3.1 堆的概念

普通的二叉樹是不適合用數(shù)組來(lái)存儲(chǔ)的,因?yàn)榭赡軙?huì)存在大量的空間浪費(fèi),而完全二叉樹更適合用順序存儲(chǔ),順序存儲(chǔ)的隨機(jī)訪問(wèn)特性,會(huì)右巨大的優(yōu)點(diǎn)。我們通常把堆(是一種完全二叉樹)采用順序存儲(chǔ),需要注意的是這里的堆和操作系統(tǒng)虛擬進(jìn)程地址空間中的堆是兩回事,一個(gè)是數(shù)據(jù)結(jié)構(gòu)一個(gè)是OS中管理內(nèi)存的一塊區(qū)域分段。
堆分為大根堆和小根堆。
大根堆:父親結(jié)點(diǎn)>=孩子結(jié)點(diǎn)
小根堆:父親結(jié)點(diǎn)<=孩子結(jié)點(diǎn)

在這里插入圖片描述

上面是堆的邏輯結(jié)構(gòu),下面是物理結(jié)構(gòu)

3.2 堆的實(shí)現(xiàn)

首先構(gòu)建堆我們要了解一個(gè)算法,叫向下調(diào)整算法。我們以小根堆為例,我們把圖示的完全二叉樹構(gòu)建為小堆,這個(gè)二叉樹有個(gè)條件是根結(jié)點(diǎn)的兩個(gè)子樹都是小堆才可以進(jìn)行向下調(diào)整算法。

在這里插入圖片描述

3.2.1 向下調(diào)整算法

在這里插入圖片描述

根結(jié)點(diǎn)的左右子樹都是小堆,根結(jié)點(diǎn)27和左右子樹的根結(jié)點(diǎn)較小的那一個(gè)交換位置,然后依次進(jìn)行,直到葉子結(jié)點(diǎn)。就把最小的15結(jié)點(diǎn)上浮到堆頂?shù)奈恢谩_@個(gè)算法的前提是根節(jié)點(diǎn)的左右子樹都是小堆,如果我們想把任意的的數(shù)組構(gòu)建成小堆顯然不滿足條件。在下面我們介紹把任意數(shù)組構(gòu)建成小堆的辦法。
向下調(diào)整算法的代碼如下:

void AdjustDown(HeapDataType* a, int n, int root)
{
	//父子下標(biāo)的初始化
	int parent = root;
	int child = parent * 2 + 1;
	//循環(huán)向下調(diào)整,把最小值(或者最大值浮到堆頂)
	while (child < n)
	{
		//選出左右孩子中較小的孩子,作為child,child+1 < n保證下標(biāo)不能越界
		if (child+1 < n && a[child + 1] < a[child])
		{
			++child;
		}
		//父親比孩子小二者交換位置,并更新迭代孩子的位置
		if (a[child] < a[parent])
		{
			Swap(&a[child], &a[parent]);
			//迭代parent child
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;//當(dāng)孩子 >= 父親的時(shí)候,滿足小堆的條件,跳出循環(huán)節(jié)課,否則就會(huì)死循環(huán)
		}
	}
}

3.2.2 堆的構(gòu)建

在這里插入圖片描述

把給定的數(shù)據(jù)化成完全二叉樹的邏輯結(jié)構(gòu)如上圖所示,這個(gè)數(shù)組(二叉樹)顯然不能滿足向下調(diào)整算法的理想條件,所以我們把問(wèn)題拆分,比如你先思考下這個(gè)問(wèn)題,把左子樹和右子樹全部構(gòu)建成小堆不就滿足條件了嘛,但是子樹的左右子不是小堆怎么辦呢,那么同樣的道理,把它也構(gòu)建成子樹就可以了,那么我們可以從葉子結(jié)點(diǎn)向上每個(gè)結(jié)點(diǎn)都執(zhí)行一邊向下調(diào)整算法不就可以了嘛。其實(shí)我們不必從葉子結(jié)點(diǎn)開始因?yàn)槿~子結(jié)點(diǎn)沒(méi)有子樹其實(shí)都可以看成是小堆或者大堆。所以從第一個(gè)非葉子結(jié)點(diǎn)開始調(diào)整即可。

在這里插入圖片描述

也就是從圖中紫色圈圈畫出來(lái)的那個(gè)開始調(diào)整即可,直到根結(jié)點(diǎn)93,就會(huì)把一個(gè)數(shù)組構(gòu)建成小堆(大堆)。

for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(a, n, i);
	}

(n-1-1)/2為第一非葉子結(jié)點(diǎn)下標(biāo)。

3.2.3 堆排序

有了上面的知識(shí)做鋪墊,堆排序就可以很好的理解了。
1、把數(shù)組構(gòu)建成大堆或者小堆,位于堆頂?shù)臄?shù)據(jù)就是最大值或者最小值,把堆頂數(shù)據(jù)和最后的結(jié)點(diǎn)位置的數(shù)據(jù)(數(shù)組元素最后一個(gè))互換。然后對(duì)前n-1個(gè)結(jié)點(diǎn)重新向下調(diào)整為大堆或者小堆。直到剩下最后一個(gè)根節(jié)點(diǎn)就排序完成。
只是要升序:構(gòu)建大堆,要降序:構(gòu)建小堆,你細(xì)細(xì)品。
代碼如下:

//堆排序
void HeapSort(int* a, int n)
{
	//堆排序的第一步就是構(gòu)建堆,構(gòu)建堆的時(shí)間復(fù)雜度是O(n),此時(shí)是小堆
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(a, n, i);
	}
	//如果是升序,構(gòu)建大堆
	//如果是降序,構(gòu)建小堆
	//是反著的,因?yàn)橐妥詈笠粋€(gè)進(jìn)行交換
	int end = n - 1;
	while (end>0)
	{
		//把堆頂(最小或者最大)和最后的的元素交換,然后從0到n-2繼續(xù)向下調(diào)整
		//把次?。ù未螅┑脑匾策x出來(lái),直到剩最后一個(gè),堆排序完成
		Swap(&a[0], &a[end]);
		AdjustDown(a, end, 0);
		--end;
	}
}

### 3.2.4 堆的的增刪查改
聲明:
```c
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <string.h>
#include <windows.h>

typedef int HeapDataType;

typedef struct Heap
{
	HeapDataType* arr;
	int size;
	int capacity;
} Heap;

//堆的初始化,內(nèi)部有堆的創(chuàng)建
void HeapInit(Heap* php, HeapDataType* a, int n);
//堆的銷毀
void HeapDestory(Heap* php);
//堆的插入數(shù)據(jù)
void HeapPush(Heap* php, HeapDataType x);
//堆的刪除數(shù)據(jù)
void HeapPop(Heap* php);
//獲取堆頂數(shù)據(jù)
HeapDataType HeapTop(Heap* php);
//對(duì)傳入的數(shù)組內(nèi)的數(shù)據(jù)進(jìn)行堆排序
void HeapSort(int* a, int n);

定義:

#include "Heap.h"
//堆是一種完全二叉樹,用順序存儲(chǔ)(數(shù)組)比較好
//用于交換兩個(gè)數(shù)據(jù)
void Swap(HeapDataType* n1, HeapDataType* n2)
{
	HeapDataType temp = *n1;
	*n1 = *n2;
	*n2 = temp;
}
//向下調(diào)整算法___老重要了,這是理解堆排序和topk問(wèn)題以及堆這里相關(guān)題的基礎(chǔ)
//向下調(diào)整結(jié)束的情況有兩個(gè)一個(gè)是a[parent]<a[child],另一個(gè)是從堆頂?shù)綌?shù)組結(jié)束全部比較完
void AdjustDown(HeapDataType* a, int n, int root)
{
	//父子下標(biāo)的初始化
	int parent = root;
	int child = parent * 2 + 1;
	//循環(huán)向下調(diào)整,把最小值(或者最大值浮到堆頂)
	while (child < n)
	{
		//選出左右孩子中較小的孩子,作為child,child+1 < n保證下標(biāo)不能越界
		if (child+1 < n && a[child + 1] < a[child])
		{
			++child;
		}
		//父親比孩子小二者交換位置,并更新迭代孩子的位置
		if (a[child] < a[parent])
		{
			Swap(&a[child], &a[parent]);
			//迭代parent child
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;//當(dāng)孩子 >= 父親的時(shí)候,滿足小堆的條件,跳出循環(huán)節(jié)課,否則就會(huì)死循環(huán)
		}
	}
}
//向上調(diào)整算法
//用在HeapPush中
void AdjustUp(HeapDataType* a, int n, int child)
{
	int parent = (child - 1) / 2;
	while (child>0)
	{
		if (a[child] < a[parent])
		{
			Swap(&a[child], &a[parent]);
			//迭代
			child = parent;
			parent = (child - 1) / 2;
		}
		else
		{
			break;
		}
	}
}
//堆的初始化,內(nèi)部有堆的創(chuàng)建
void HeapInit(Heap* php, HeapDataType* a, int n)
{
	HeapDataType* temp = (HeapDataType*)malloc(sizeof(HeapDataType)*n);
	if (temp)
	{
		php->arr = temp;
		
	}
	else
	{
		printf("內(nèi)存申請(qǐng)失敗!");
		exit(-1);
	}
	//將傳進(jìn)來(lái)的數(shù)組拷貝給malloc出來(lái)的空間,用來(lái)后續(xù)的堆的創(chuàng)建,刪除,插入數(shù)據(jù)等操作
	memcpy(php->arr, a, sizeof(HeapDataType)*n);
	php->size = n;
	php->capacity = n;

	//把拷貝進(jìn)來(lái)的數(shù)組,構(gòu)建成堆
	//從倒數(shù)第一個(gè)非葉子節(jié)點(diǎn)進(jìn)行構(gòu)建(直接把數(shù)組畫成一個(gè)完全二叉樹可以直接由圖得到第一個(gè)
	//非葉子節(jié)點(diǎn)的下標(biāo))
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(php->arr, php->size, i);
	}
}
//堆排序
void HeapSort(int* a, int n)
{
	//堆排序的第一步就是構(gòu)建堆,構(gòu)建堆的時(shí)間復(fù)雜度是O(n),此時(shí)是小堆
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(a, n, i);
	}
	//如果是升序,構(gòu)建大堆
	//如果是降序,構(gòu)建小堆
	//是反著的,因?yàn)橐妥詈笠粋€(gè)進(jìn)行交換
	int end = n - 1;
	while (end>0)
	{
		//把堆頂(最小或者最大)和最后的的元素交換,然后從0到n-2繼續(xù)向下調(diào)整
		//把次小(次大)的元素也選出來(lái),直到剩最后一個(gè),堆排序完成
		Swap(&a[0], &a[end]);
		AdjustDown(a, end, 0);
		--end;
	}
}

//銷毀堆
void HeapDestory(Heap* php)
{
	assert(php);
	free(php->arr);
	php->arr = NULL;
	php->capacity = php->size = 0;
}

//堆的插入數(shù)據(jù)
void HeapPush(Heap* php, HeapDataType x)
{
	assert(php);
	if (php->size == php->capacity)
	{
		php->capacity *= 2;
		HeapDataType* tmp = (HeapDataType*)realloc(php->arr, sizeof(HeapDataType)*php->capacity);
		if (tmp)
		{
			php->arr = tmp;
		}
		else
		{
			printf("擴(kuò)容失??!\n");
			exit(-1);
		}
	}
	php->arr[php->size++] = x;
	AdjustUp(php->arr, php->size, php->size - 1);
}

//堆的刪除數(shù)據(jù),要?jiǎng)h除堆頂?shù)臄?shù)據(jù)
void HeapPop(Heap* php)
{
	assert(php);
	assert(php->size > 0);
	Swap(&php->arr[0], &php->arr[php->size - 1]);
	php->size--;
	AdjustDown(php->arr, php->size, 0);
}

//獲取堆頂數(shù)據(jù)
HeapDataType HeapTop(Heap* php)
{
	assert(php);
	assert(php->size > 0);
	return php->arr[0];
}

看這里的小伙伴可以自己下去測(cè)試一下代碼哦。

到此這篇關(guān)于C++實(shí)現(xiàn)二叉樹及堆的示例代碼的文章就介紹到這了,更多相關(guān)C++實(shí)現(xiàn)二叉樹及堆內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++ const關(guān)鍵字的實(shí)例用法

    C++ const關(guān)鍵字的實(shí)例用法

    在本篇文章里小編給大家整理的是一篇關(guān)于C++ const關(guān)鍵字的實(shí)例用法,需要的朋友們可以學(xué)習(xí)下。
    2020-02-02
  • 關(guān)于C++一些特性的探究

    關(guān)于C++一些特性的探究

    下面小編就為大家?guī)?lái)一篇關(guān)于C++一些特性的探究。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-06-06
  • C語(yǔ)言實(shí)現(xiàn)字符轉(zhuǎn)unix時(shí)間戳的簡(jiǎn)單實(shí)例

    C語(yǔ)言實(shí)現(xiàn)字符轉(zhuǎn)unix時(shí)間戳的簡(jiǎn)單實(shí)例

    下面小編就為大家?guī)?lái)一篇C語(yǔ)言實(shí)現(xiàn)字符轉(zhuǎn)unix時(shí)間戳的簡(jiǎn)單實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-06-06
  • C++實(shí)現(xiàn)LeetCode(83.移除有序鏈表中的重復(fù)項(xiàng))

    C++實(shí)現(xiàn)LeetCode(83.移除有序鏈表中的重復(fù)項(xiàng))

    這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(83.移除有序鏈表中的重復(fù)項(xiàng)),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下
    2021-07-07
  • C/C++深入講解內(nèi)存管理

    C/C++深入講解內(nèi)存管理

    本章主要介紹C語(yǔ)言與C++的內(nèi)存管理,以C++的內(nèi)存分布作為引入,介紹C++不同于C語(yǔ)言的內(nèi)存管理方式(new?delete對(duì)比?malloc?free),感興趣的朋友來(lái)看看吧
    2022-05-05
  • c++代碼各種注釋示例詳解

    c++代碼各種注釋示例詳解

    大家好,本篇文章主要講的是c++代碼各種注釋示例詳解,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12
  • 淺談C++STL之雙端隊(duì)列容器

    淺談C++STL之雙端隊(duì)列容器

    deque雙端隊(duì)列容器與vector很類似,采用線性表順序存儲(chǔ)結(jié)構(gòu)。但與vector區(qū)別,deque采用分塊的線性存儲(chǔ)結(jié)構(gòu)來(lái)存儲(chǔ)數(shù)據(jù),每塊的大小一般為512B,將之稱為deque塊,所有的deque塊使用一個(gè)map塊進(jìn)行管理,每個(gè)map數(shù)據(jù)項(xiàng)記錄各個(gè)deque塊的首地址。
    2021-06-06
  • C++使用遞歸方法求n階勒讓德多項(xiàng)式完整實(shí)例

    C++使用遞歸方法求n階勒讓德多項(xiàng)式完整實(shí)例

    這篇文章主要介紹了C++使用遞歸方法求n階勒讓德多項(xiàng)式,涉及C++遞歸算法與浮點(diǎn)數(shù)運(yùn)算的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2016-05-05
  • Qt實(shí)現(xiàn)自定義驗(yàn)證碼輸入框控件的方法

    Qt實(shí)現(xiàn)自定義驗(yàn)證碼輸入框控件的方法

    本文主要介紹了Qt實(shí)現(xiàn)自定義驗(yàn)證碼輸入框控件的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-04-04
  • C語(yǔ)言菜鳥基礎(chǔ)教程之判斷

    C語(yǔ)言菜鳥基礎(chǔ)教程之判斷

    C語(yǔ)言判斷結(jié)構(gòu)要求程序員指定一個(gè)或多個(gè)要評(píng)估或測(cè)試的條件,以及條件為真時(shí)要執(zhí)行的語(yǔ)句(必需的)和條件為假時(shí)要執(zhí)行的語(yǔ)句(可選的)
    2017-10-10

最新評(píng)論