溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

c++顯式棧如何實現遞歸

發(fā)布時間:2022-01-07 17:35:27 來源:億速云 閱讀:211 作者:柒染 欄目:開發(fā)技術

本篇文章為大家展示了c++顯式棧如何實現遞歸,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。

前言

在大學的課上老師有教過,也就是用循環(huán)來實現遞歸,現在自己回顧一下并且做一下記錄。

1. 遞歸

假設有函數A, 和函數B, 函數B是一個遞歸函數, 函數A調用函數B。
這個遞歸的過程分為:

函數A調用函數B,函數A將數據傳給函數B。此時進入到函數B內部,函數B通過傳參拿到函數A傳過來的數據。執(zhí)行本次調用的操作將新的數據作為參數傳入函數B(遞歸過程, 內部再次執(zhí)行2~3步驟,以此類推)。退出遞歸結束。

2. 顯式棧實現的思路

由上面的過程可以不難看出,遞歸的過程遵循 后進后出 這樣的一個規(guī)律。那么就很容易聯想到具有同樣特征的棧這樣一個數據結構。這里給出顯式棧實現遞歸的思路:
假設已經申請了一個stack的容器,

首先將初始數據壓棧,這個類似于遞歸過程中的函數A最開始調用函數B時將數據傳入的操作。接下來是循環(huán)操作,條件是true,也就是死循環(huán), 這個類似于函數B內部一直遞歸調用,至于什么時候結束取決于什么時候遇到遞歸出口;在這個死循環(huán)里應該在每次循環(huán)時進行一次條件判定,這個條件判定相當于遞歸函數中決定什么時候返回的條件判定;接下來進到循環(huán)內部,首先取棧頂數據出來,這類似函數B內部取到了傳參執(zhí)行 本次的循環(huán)的關鍵操作,處理數據的任務將新的數據壓棧,這部分相當于將新的數據作為參數傳入函數B如果觸發(fā)了循環(huán)退出條件,則退出循環(huán)

3. 代碼解析

上面說了具體思路,現在用代碼來說明,首先上遞歸的寫法, 用遍歷二叉樹作為例子。

#include<iostream>
using namespace std;
class Node
{
public:
	int value;
	Node* left_child;
	Node* right_child;
	Node(int data)
	{
		this->data = data;
		this->left_child = nullptr;
		this->right_child = nullptr;
	}
};

void B(Node* node)
{
	//這個時候已經經歷了步驟2, 函數B拿到了數據root
	// 步驟3,執(zhí)行本次遞歸調用的關鍵操作
	cout << node->data<< endl; 
	// 步驟4,拿到新的數據root->left_child和root->right_child
	//調用函數B
	if (node->left_child) B(node->left_child);
	if (node->right_child) B(node->right_child);
	//步驟5,遞歸結束
}

void A()
{
	Node root(10);  //模擬一顆樹
	B(&root); //步驟1,傳參
}

int main()
{
	A();
}
以上步驟3和步驟4的順序不是固定的,他們順序的不同各自構成了不同的樹遍歷方法(先序,中序,后序遍歷)。接下來是顯式棧實現的寫法
#include<iostream>
#include<stack>
using namespace std;
class Node
{
public:
	int value;
	Node* left_child;
	Node* right_child;
	Node(int data)
	{
		this->data = data;
		this->left_child = nullptr;
		this->right_child = nullptr;
	}
};

int main()
{
	Node root(10);  //模擬一顆樹
	stack<Node*> m_stack;
	m_stack.push(&root); //步驟1,將根節(jié)點壓棧, 相當于函數A調用函數B時傳參
	while(true)
	{
		if (m_stack.empty())
		{
			break; 
			//這里相當于步驟5,判定循環(huán)結束條件, 也可以寫到while條件上
			//為了思路更清晰,所以寫在循環(huán)里面,也更好跟遞歸版本進行比較
		}
		//步驟2,取棧頂元素
		Node* current_node = m_stack.top();
		m_stack.pop();
		//步驟3,執(zhí)行本次循環(huán)的關鍵操作
		cout << current_node->data<< endl;
		//步驟4, 拿到新的數據并且壓棧
		if (current_node->left_child)
			m_stack.push(current_node->left_child);
		if (current_node->right_child)
			m_stack.push(current_node->right_child);
	}
}
顯式棧實現的版本里,有一個細節(jié)就是取完棧頂數據之后需要將棧頂的數據出棧,這樣才能使用棧是否空來判斷遞歸出口。

上述內容就是c++顯式棧如何實現遞歸,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

c++
AI