亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C語言進(jìn)階練習(xí)二叉樹的遞歸遍歷

 更新時(shí)間:2022年06月24日 09:24:15   作者:配的上了嗎  
樹是一種重要的非線性數(shù)據(jù)結(jié)構(gòu),直觀地看,它是數(shù)據(jù)元素(在樹中稱為結(jié)點(diǎn))按分支關(guān)系組織起來的結(jié)構(gòu),很象自然界中的樹那樣。樹結(jié)構(gòu)在客觀世界中廣泛存在,如人類社會(huì)的族譜和各種社會(huì)組織機(jī)構(gòu)都可用樹形象表示,本篇介紹二叉樹的遞歸與非遞歸遍歷的方法

二叉樹的前中后序遍歷

所謂二叉樹遍歷(Traversal)是按照某種特定的規(guī)則,依次對(duì)二叉樹中的節(jié)點(diǎn)進(jìn)行相應(yīng)的操作,并且每個(gè)節(jié)點(diǎn)只操作一次。訪問結(jié)點(diǎn)所做的操作依賴于具體的應(yīng)用問題。

遍歷 是二叉樹上最重要的運(yùn)算之一,也是二叉樹上進(jìn)行其它運(yùn)算的基礎(chǔ)。

按照規(guī)則,二叉樹的遍歷有:前序/中序/后序的遞歸結(jié)構(gòu)遍歷:

1. 前序遍歷(Preorder Traversal 亦稱先序遍歷)——訪問根結(jié)點(diǎn)的操作發(fā)生在遍歷其左右子樹之前。

2. 中序遍歷(Inorder Traversal)——訪問根結(jié)點(diǎn)的操作發(fā)生在遍歷其左右子樹之中(間)。

3. 后序遍歷(Postorder Traversal)——訪問根結(jié)點(diǎn)的操作發(fā)生在遍歷其左右子樹之后。

前序遍歷示意圖

// 二叉樹前序遍歷
void PreOrder(BTNode* root)
{
	if (root == nullptr)
	{
		cout << "# ";
		return;       // 空的話結(jié)束遞歸,輸出#來表示這是一個(gè)空結(jié)點(diǎn)
	}
	cout << root->data << " ";
	PreOrder(root->left);
	PreOrder(root->right);
}
// 二叉樹中序遍歷
void InOrder(BTNode* root)
{
	if (root == nullptr)
	{
		cout << "# ";
		return;
	}
	InOrder(root->left);
	cout << root->data << " ";
	InOrder(root->right);
}
// 二叉樹后序遍歷
void PostOrder(BTNode* root)
{
	if (root == nullptr)
	{
		cout << "# ";
		return;
	}
	PostOrder(root->left);
	PostOrder(root->right);
	cout << root->data << " ";
}

其實(shí)前中后序遍歷的區(qū)別,只是在于,對(duì)這個(gè)結(jié)點(diǎn)進(jìn)行某些操作的時(shí)機(jī),是在遍歷其左右子樹之前,之中還是之后。這個(gè)操作由具體要解決的問題決定。上方例子中是以打印為例。并且左子樹的遍歷通常都在其右子樹遍歷之前。

就是,把每個(gè)非空的根節(jié)點(diǎn)看作一個(gè)二叉樹,進(jìn)行同樣的操作就是二叉樹的遞歸遍歷。這些二叉樹的遞歸遍歷之間有一定的順序。遞歸的結(jié)束條件是,這個(gè)結(jié)點(diǎn)為空,為空則不進(jìn)行下一步遞歸。形如結(jié)點(diǎn)3,它的左右子樹為空,在這里結(jié)束此處的遞歸,然后返回給上一層。

遍歷二叉樹求二叉樹的結(jié)點(diǎn)個(gè)數(shù)

int count = 0;
void TreeSize1(BTNode* root)
{
	if (root == nullptr)
		return;
	++::count;
	TreeSize1(root->left);
	TreeSize1(root->right);
}
int TreeSize2(BTNode* root)
{
	if (root == NULL)
		return 0;
	return 1 + TreeSize2(root->left) + TreeSize2(root->right);
}

兩種遍歷方式,顯然第二種更好,其實(shí)可以直接從遞歸,然后第一次遞歸到底部,開始思考這個(gè)計(jì)算過程。

如下圖,遞歸至3結(jié)點(diǎn)時(shí),3結(jié)點(diǎn)返回1+leftsize+rightsize 顯然其左右返回0,所以3結(jié)點(diǎn)返回1,2結(jié)點(diǎn)的左返回1,然后求2結(jié)點(diǎn)的右個(gè)數(shù),顯然返回0,2結(jié)點(diǎn)返回給1結(jié)點(diǎn)2,至此,1結(jié)點(diǎn)的左返回2,然后求1結(jié)點(diǎn)的右,4結(jié)點(diǎn)的左返回1,右返回1,4結(jié)點(diǎn)返回給1結(jié)點(diǎn)3,所以最終1結(jié)點(diǎn)返回1+2+3 = 6。 當(dāng)然,5和6結(jié)點(diǎn)都是求左右加1的這么一個(gè)步驟。

遍歷二叉樹求二叉樹的葉子結(jié)點(diǎn)個(gè)數(shù)

int LeafTreeNode(BTNode* root)
{
	if (root == nullptr)
	{
		return 0;
	}
	else if (root->left == NULL && root->right == NULL)
	{
		return 1;
	}
	else
	{
		return LeafTreeNode(root->left) + LeafTreeNode(root->right);
	}
}

也是非常好理解的,不是葉子,不是空,代表其左右子樹至少有一個(gè)子樹不為空,則返回其左右子樹的葉子節(jié)點(diǎn)個(gè)數(shù),典型的分治思想。如下圖,對(duì)于1結(jié)點(diǎn),返回其左右子樹的葉子節(jié)點(diǎn)個(gè)數(shù)之和即可,空返回0是防止結(jié)點(diǎn)2的右子樹,這樣2結(jié)點(diǎn)才能正確地返回給1結(jié)點(diǎn)1。

遞歸求二叉樹的第k層的結(jié)點(diǎn)個(gè)數(shù)

int TreeKLevel(BTNode* root, int k)   //求第k層
{
	if (root == NULL)
		return 0;
	if (k == 1)
		return 1;
	return TreeKLevel(root->left, k - 1) + TreeKLevel(root->right, k - 1);
}

遞歸的結(jié)束條件是,當(dāng)這個(gè)結(jié)點(diǎn)是空,不管k是不是1,都結(jié)束遞歸,另一個(gè)情況就是,此結(jié)點(diǎn)k是1,且不是空,表示這個(gè)結(jié)點(diǎn)就是所求的目標(biāo)節(jié)點(diǎn),無論結(jié)點(diǎn)下方是否還有結(jié)點(diǎn)都結(jié)束遞歸。

求二叉樹中data為x的結(jié)點(diǎn)

BTNode* TreeFind(BTNode* root, BTDataType x)
{
	if (root == NULL)
		return NULL;
	if (root->data == x)
		return root;
	BTNode* retleft = TreeFind(root->left, x);
	if (retleft)
		return retleft;
	BTNode* retright = TreeFind(root->right, x);
	if (retright)
		return retright;
	return NULL;
}

典型的前序遍歷,每到一個(gè)根節(jié)點(diǎn),先判斷是否為空,非空則判斷是否為目標(biāo)結(jié)點(diǎn),不是的話,就先去其左子樹找,左子樹沒有則去右子樹找,右子樹沒有則表示這顆二叉樹中無目標(biāo)節(jié)點(diǎn),返回NULL。這個(gè)流程對(duì)于每顆二叉樹都適用。

因?yàn)橹皇乔蟪鲆粋€(gè)值為x的結(jié)點(diǎn),所以若當(dāng)前結(jié)點(diǎn)是x,或者其左子樹有x,都會(huì)結(jié)束遞歸。不難理解。

求二叉樹的深度

int  TreeDepth(BTNode* root)
{
	if (root == NULL)
		return 0;
	int leftdepth = TreeDepth(root->left);
	int rightdepth = TreeDepth(root->right);
	return 1 + leftdepth > rightdepth ? leftdepth : rightdepth;
}

對(duì)于1結(jié)點(diǎn),返回1+左右子樹更深的那個(gè)子樹,其實(shí)完全可以遞歸至3然后往回思考,注意每一個(gè)結(jié)點(diǎn)都是遞歸求左子樹的深度之后才會(huì)遞歸求右子樹的深度。

到此這篇關(guān)于C語言進(jìn)階練習(xí)二叉樹的遞歸遍歷的文章就介紹到這了,更多相關(guān)C語言二叉樹內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C語言 詳細(xì)講解#pragma的使用方法

    C語言 詳細(xì)講解#pragma的使用方法

    #pragma 指令對(duì)每個(gè)編譯器給出了一個(gè)方法,在保持與C和C++語言完全兼容的情況下,給出主機(jī)或操作系統(tǒng)專有的特征。依據(jù)定義,編譯指示是機(jī)器或操作系統(tǒng)專有的, 且對(duì)于每個(gè)編譯器都是不同的
    2022-04-04
  • C++?Boost?ProgramOptions超詳細(xì)講解

    C++?Boost?ProgramOptions超詳細(xì)講解

    Boost是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱。Boost庫是一個(gè)可移植、提供源代碼的C++庫,作為標(biāo)準(zhǔn)庫的后備,是C++標(biāo)準(zhǔn)化進(jìn)程的開發(fā)引擎之一,是為C++語言標(biāo)準(zhǔn)庫提供擴(kuò)展的一些C++程序庫的總稱
    2022-11-11
  • C++11右值引用和轉(zhuǎn)發(fā)型引用教程詳解

    C++11右值引用和轉(zhuǎn)發(fā)型引用教程詳解

    這篇文章主要介紹了C++11右值引用和轉(zhuǎn)發(fā)型引用教程詳解,需要的朋友可以參考下
    2018-03-03
  • C語言手寫集合List的示例代碼

    C語言手寫集合List的示例代碼

    數(shù)組長(zhǎng)度是固定的,那么在很多時(shí)候我們并不知道到底有多少數(shù)據(jù)需要存儲(chǔ),這時(shí)候我么就需要一個(gè)可變長(zhǎng)度的數(shù)組來進(jìn)行存儲(chǔ),在C語言中需要我們自己進(jìn)行定義,我們稱為集合。本文將用C語言實(shí)現(xiàn)手寫集合,需要的可以參考一下
    2022-08-08
  • C++網(wǎng)絡(luò)編程詳細(xì)講解

    C++網(wǎng)絡(luò)編程詳細(xì)講解

    計(jì)算機(jī)是通過TCP/IP協(xié)議進(jìn)行互聯(lián)從而進(jìn)行通信的,為了把復(fù)雜的TCP/IP協(xié)議隱藏起來,更方便的實(shí)現(xiàn)計(jì)算機(jī)中兩個(gè)程序進(jìn)行通信,引出了socket這個(gè)概念
    2022-10-10
  • c++ 遞歸鎖的使用示例代碼

    c++ 遞歸鎖的使用示例代碼

    這篇文章主要介紹了c++ 遞歸鎖的使用示例代碼,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-08-08
  • linux c多線程編程實(shí)例代碼

    linux c多線程編程實(shí)例代碼

    這篇文章主要介紹了linux系統(tǒng)中的c多線程編程實(shí)例,大家可以參考使用以下代碼
    2013-11-11
  • C++學(xué)習(xí)貝葉斯分類器實(shí)現(xiàn)手寫數(shù)字識(shí)別示例解析

    C++學(xué)習(xí)貝葉斯分類器實(shí)現(xiàn)手寫數(shù)字識(shí)別示例解析

    這篇文章主要介紹了在C++學(xué)習(xí)中如何采用貝葉斯分類器來實(shí)現(xiàn)手寫數(shù)字識(shí)別的示例及解析有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2021-10-10
  • C語言實(shí)現(xiàn)掃雷附完整代碼

    C語言實(shí)現(xiàn)掃雷附完整代碼

    本文詳細(xì)講解了C語言實(shí)現(xiàn)掃雷并附完整代碼,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-11-11
  • C語言 動(dòng)態(tài)內(nèi)存開辟常見問題解決與分析流程

    C語言 動(dòng)態(tài)內(nèi)存開辟常見問題解決與分析流程

    動(dòng)態(tài)內(nèi)存是相對(duì)靜態(tài)內(nèi)存而言的。所謂動(dòng)態(tài)和靜態(tài)就是指內(nèi)存的分配方式。動(dòng)態(tài)內(nèi)存是指在堆上分配的內(nèi)存,而靜態(tài)內(nèi)存是指在棧上分配的內(nèi)存
    2022-03-03

最新評(píng)論