【二叉树构建与遍历2】后序遍历+中序遍历构建一个二叉树并输出先序遍历 C++实现

思路:

先来一个例子:

后序遍历序列为:XEDGAF

中序遍历序列为:XDEFAG

要根据后序序列和中序序列确定这个二叉树,通用的步骤为:

1.根据后序序列的最后一位确定这棵树的根;

2.在中序序列中找到根的所在的位置,根的左边就是该树的左子树的节点,根的右边就是该树的右子树的节点;

3.根据树的左子树节点和右子树节点在后序序列中分别找到对应的子串;

4.对3中找到的两个子串分别重复1 2 3步,左子树节点用于构建左子树,右子树节点用于构建右子树,直到所有的节点都用于构建这棵树。

示例:

根据以上的案例走一遍:

1.由后序遍历序列为:XDEGAF可知,树的根节点为F;

2.在中序遍历序列XDEFAG找到F的索引为3,左边XDE就是该树的左子树的节点,右边AG就是该树的右子树的节点;

3.根据树的左子树节点XDE和右子树节点AG在后序序列中分别找到对应的子串,分别为:XDE和GA;

4.对XDE和GA分别重复步骤1 2 3,XDE用于构建左子树,GA用于构建右子树,直到所有的节点都用于构建树。

每执行完第一轮步骤1 2 3,所用的根节点已经用于构建树了,后续就不用再考虑了。例如,执行完第一轮步骤1 2 3,根节点F已经用过了,后续就不用再考虑了。

分析:

相信思路都很明确,步骤大概也明白了,接下来代码实现中一个非常注重细节的地方就是递归构建左子树和右子树的部分,再详细说就是在递归调用构建树的函数中,我们要传入的两个参数应该怎么确定。

本次主要介绍利用后序遍历序列中序遍历序列 构建一个二叉树并输出后序遍历的方法,我们在递归调用构建树的函数中,我们要传入的两个参数当然就是子树的后序遍历序列中序遍历序列。创建左子树时就传入左子树的后序遍历序列和中序遍历序列,创建右子树时就传入右子树的后序遍历序列和中序遍历序列。

以下根据以上案例详细分析:对于:

cpp 复制代码
索         引:012345
先序遍历序列为:XEDGAF

中序遍历序列为:XDEFAG

将以上两个遍历序列分别命名为字符串s1,s2,即s1 = "XEDGAF", s2 = "XDEFAG"。根据s1可知树的根为F,在s2中寻找到F的索引(定义为pos)为3。根据s2可得左子树包括的字符有:XDE(由s2.substr(0, pos)获得),右子树包括的字符有:AG(由s2.substr(pos+1)获得),在s1中对应的字符串分别为:XED(由s1.substr(0, pos)获得)和GA(由s1.substr(pos, str2.size() - pos - 1)获得)。根据以上获得的四个参数,可以递归创建左子树和右子树。

源代码:

cpp 复制代码
//根据后序遍历和中序遍历确定一个二叉树
// 二叉树节点结构定义
struct TreeNode {
	char data;
	TreeNode* leftChild;
	TreeNode* rightChild;
	TreeNode(char c) : data(c), leftChild(NULL), rightChild(NULL) {}
};

// 根据后序遍历和中序遍历构建二叉树
TreeNode* Build(string str1, string str2) {
	if (str1.size() == 0) {
		return NULL;
	}
	// 取先序遍历的第一个字符作为根节点
	char c = str1[str1.size() - 1];
	// 在中序遍历中找到根节点的位置
	int pos = str2.find(c);
	// 创建根节点
	TreeNode* root = new TreeNode(c);
	// 递归构建左子树
	root->leftChild = Build(str1.substr(0, pos), str2.substr(0, pos));
	// 递归构建右子树
	root->rightChild = Build(str1.substr(pos, str2.size() - pos - 1), str2.substr(pos+1));

	return root;
}

//先序遍历
void preOrder(TreeNode* root) {
	if (root == NULL) {
		return;
	}
	//输出当前根节点的值
	cout << root->data;
	//先遍历左子树
	preOrder(root->leftChild);
	//再遍历右子树
	preOrder(root->rightChild);

	return;
}

int main()
{
	string s1, s2;
	while (getline(cin, s1)) {
		getline(cin, s2);
		//根据后序遍历和中序遍历构建该二叉树
		TreeNode* root = Build(s1, s2);
		//先序遍历该二叉树
        cout<<"先序遍历的结果为:";
		preOrder(root);
		cout << endl;
	}

	return 0;
}

示例运行结果:

相关推荐
是十一月末9 分钟前
opencv实现KNN算法识别图片数字
人工智能·python·opencv·算法·k-近邻算法
袖清暮雨18 分钟前
5_SparkGraphX讲解
大数据·算法·spark
Tisfy27 分钟前
LeetCode 3218.切蛋糕的最小总开销 I:记忆化搜索(深度优先搜索DFS)
算法·leetcode·深度优先·题解·记忆化搜索
pl002035 分钟前
C++运算符重载实例
开发语言·c++·运算符重载·单目运算符·双目运算符·流运算符
张明奇-琦玉42 分钟前
Boost之log日志使用
linux·服务器·算法
煤泥做不到的!43 分钟前
挑战一个月基本掌握C++(第十二天)了解命名空间,模板,预处理器
开发语言·c++
青春男大1 小时前
java队列--数据结构
java·开发语言·数据结构·学习·eclipse
XZHOUMIN1 小时前
【MFC】多工具栏如何保存状态(续)
c++·mfc
界面开发小八哥1 小时前
MFC扩展库BCGControlBar Pro v36.0 - 可视化管理器等全新升级
c++·mfc·bcg·界面控件·ui开发
Kai HVZ1 小时前
《机器学习》——利用OpenCV库中的KNN算法进行图像识别
opencv·算法·机器学习