【数据结构·考研】后序线索二叉树

news2024/9/29 1:19:37

之前我们介绍了中序线索二叉树的构造、遍历、寻找前趋和寻找后继;先序线索二叉树的构造、遍历、寻找后继。先序线索二叉树无法解决寻找前趋的任务,中序线索二叉树既可以寻找前趋结点也可以寻找后继结点。

今天我们再来看一下后序线索二叉树的构造:

首先是结构声明:

我们将左线索和右线索初始化为0。

/*
结构声明 
*/ 
typedef struct node{
	struct node* left;
	int ltag = 0;
	char val;
	int rtag = 0;
	struct node* right;
}TreeNode,*Tree; 

后序构造线索二叉树的递归代码:

与中序构建线索二叉树的代码基本相同,把对当前结点的操作放到了遍历左右子树之后。

void PostThread(Tree& t, TreeNode* &pre){
	//pre指针指向t的中序前驱,在主函数中预设为NULL
	if(t != NULL){
		PostThread(t->left, pre); //左子树线索化
		PostThread(t->right, pre); //右子树线索化 
		if(t->left == NULL){ //建立当前结点的前驱线索 
			t->left = pre;
			t->ltag = 1;
		}
		if(pre && pre->right == NULL){ //建立前驱结点pre的后继线索 
			pre->right = t;
			pre->rtag = 1;
		}
		pre = t;
	}
} 

继续包装一下递归函数,与中序和先序不同的地方在于最后的pre指针指向根节点,所以不需要收尾。

/*
通过后序遍历建立后序线索二叉树
*/ 
void CreateThread(Tree& t){
	TreeNode* pre = NULL; //前驱指针
	if(t != NULL){
		PostThread(t,pre);
	}	
} 

后序线索二叉树访问前趋:

/*
返回结点p在线索二叉树中的后序序列下的前驱结点
如果有右孩子,则前驱是右孩子,如果没有右孩子,则前驱是左线索
*/
TreeNode* predecessor(TreeNode* p){
	if(p->rtag == 0) return p->right; 
	return p->left;
} 

后序线索二叉树不能很好的解决寻找后继。建立好后序线索二叉树之后,我们就可以根据线索和左右孩子指针以线性的时间来先序遍历后序线索二叉树了。

/*
后序线索二叉树不能很好的解决寻找后继。
这里给出一直求前驱得到的后序遍历的逆序列 
*/ 
void PostOrder(TreeNode* t){
	TreeNode*p = t;
	for(p; p!= NULL; p = predecessor(p))
		cout<< p->ltag <<" "<< p->val <<" "<< p->rtag <<" ";
}

 我们可以层次遍历一下,看看树打上标记后的样子。

/*
层次遍历
因为线索二叉树的叶子节点几乎没有了空指针,所以进队的条件应该做修改,用 ltag、rtag 来判断
*/ 
void levelOrderTraverse(Tree& t){
	if(t == NULL) return;
	queue<TreeNode*> q;
	TreeNode* p;
	q.push(t);
	while(!q.empty()){
		int width = q.size();
		for(int i = 0;i < width;i ++){
			p = q.front();
			q.pop();
			cout<<p->ltag<<" "<<p->val<<" "<<p->rtag<<"  ";
			if(p->ltag == 0) q.push(p->left);
			if(p->rtag == 0) q.push(p->right);
		}
		cout<<endl;
	}
} 

完整代码:

#include<iostream>
#include<queue>
using namespace std;

/*
结构声明 
*/ 
typedef struct node{
	struct node* left;
	int ltag = 0;
	char val;
	int rtag = 0;
	struct node* right;
}TreeNode,*Tree; 
 
 
void PostThread(Tree& t, TreeNode* &pre){
	//pre指针指向t的中序前驱,在主函数中预设为NULL
	if(t != NULL){
		PostThread(t->left, pre); //左子树线索化
		PostThread(t->right, pre); //右子树线索化 
		if(t->left == NULL){ //建立当前结点的前驱线索 
			t->left = pre;
			t->ltag = 1;
		}
		if(pre && pre->right == NULL){ //建立前驱结点pre的后继线索 
			pre->right = t;
			pre->rtag = 1;
		}
		pre = t;
	}
} 
 
/*
通过后序遍历建立后序线索二叉树
*/ 
void CreateThread(Tree& t){
	TreeNode* pre = NULL; //前驱指针
	if(t != NULL){
		PostThread(t,pre);
	}	
} 
 
/*
返回结点p在线索二叉树中的后序序列下的前驱结点
如果有右孩子,则前驱是右孩子,如果没有右孩子,则前驱是左线索
*/
TreeNode* predecessor(TreeNode* p){
	if(p->rtag == 0) return p->right; 
	return p->left;
} 


/*
后序线索二叉树不能很好的解决寻找后继。
这里给出一直求前驱得到的后序遍历的逆序列 
*/ 
void PostOrder(TreeNode* t){
	TreeNode*p = t;
	for(p; p!= NULL; p = predecessor(p))
		cout<< p->ltag <<" "<< p->val <<" "<< p->rtag <<" ";
}

 
/*
层次遍历
因为线索二叉树的叶子节点几乎没有了空指针,所以进队的条件应该做修改,用 ltag、rtag 来判断
*/ 
void levelOrderTraverse(Tree& t){
	if(t == NULL) return;
	queue<TreeNode*> q;
	TreeNode* p;
	q.push(t);
	while(!q.empty()){
		int width = q.size();
		for(int i = 0;i < width;i ++){
			p = q.front();
			q.pop();
			cout<<p->ltag<<" "<<p->val<<" "<<p->rtag<<"  ";
			if(p->ltag == 0) q.push(p->left);
			if(p->rtag == 0) q.push(p->right);
		}
		cout<<endl;
	}
} 
 
/*
先序构造二叉树
*/ 
void CreateTree(Tree& t){
	char x;
	cin>>x;
	if(x == '#') t = NULL; 
	else{
		t = new TreeNode; 
		t->val = x;  
		CreateTree(t->left); 
		CreateTree(t->right); 
	}
} 
 
int main(){
	Tree t;
	CreateTree(t);
	TreeNode* parent;
	/*
	   a b d # # e # # c f # # #
	*/
	CreateThread(t); 
	cout<<"层次遍历:"<<endl;
	levelOrderTraverse(t);
	cout<<"后序遍历的逆序列"<<endl;
	PostOrder(t); 
}

程序运行结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/398963.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

apt-get通过代理更新系统

apt-get通过代理更新系统1. 源由2. 需求3. 分析4. 步骤4.1 安装CCProxy4.2 配置CCProxy4.3 apt-get更新4.4 apt-get升级4.5 apt-get安装5. 补充资料 命令行设置代理6. 参考资料1. 源由 apt-get经典操作就是通过网络与服务器通信&#xff0c;下载相应的包来安装和更新软件。 这…

docker容器内安装gcc(trunk 最新版本)以及LLVM

1、docker内部只有wget以及git命令 项目需要&#xff0c;得更新docker容器中的gcc和LLVM版本但是由于没有预先安装apt、apt-get以及yum&#xff0c;导致很多安装过程就是鸡生蛋蛋生鸡反应。暂时没有找到合适的解决的方法&#xff0c;如果有大佬知道的话&#xff0c;欢迎留言哈…

ZeroTier实现内网穿透详细教程,无需公网IP,实现异地组网

ZeroTier实现内网穿透详细教程&#xff0c;无需公网IP&#xff0c;实现异地组网ZeroTier1.官网注册账号&#xff0c;创建自己的局域网段2.点击创建好的网络&#xff0c;进入设置界面进行设置3.下载客户端&#xff0c;安装客户端&#xff0c;然后连接到网络中4.加入网络成功后&a…

pocsuite3安装及使用

pocsuite3安装及使用简介项目地址环境配置及安装环境要求安装&#xff08;详情可以参考[https://pocsuite.org/](https://pocsuite.org/)&#xff09;使用方法运行模块加载目标参数&#xff1a;Console模式查看有哪些模块使用Telnet 弱密码模块这里以flask模板注入漏洞为例pocs…

阿里云短信验证

1.了解阿里云用户权限操作 需要通过个人账户获得 授权码&#xff08;id、密码&#xff09;&#xff0c;再通过这些信息获得服务 阿里云网址 &#xff1a;https://www.aliyun.com/ 1.登陆阿里云服务器2.进入个人账号然后点击 AccessKey 管理3.创建用户组4.添加用户组权限&…

SYSU程设c++(第三周) 对象类、类的成员、类与结构体的区别、类的静态成员

对象&类 类用于指定对象的形式&#xff0c;它包含数据的表示方法和用于处理数据的方法。 • 类中的数据和方法称为类的成员。 • 函数在一个类中也被称为类的成员。 定义一个类&#xff0c;其效果是定义一个数据类型的蓝图。它定义了类的对象包括了什么&#xff0c;以及可…

C++基础了解-08-C++运算符

C运算符 一、C 运算符 运算符是一种告诉编译器执行特定的数学或逻辑操作的符号。C 内置了丰富的运算符&#xff0c;并提供了以下类型的运算符&#xff1a; 算术运算符 关系运算符 逻辑运算符 位运算符 赋值运算符 杂项运算符 二、算术运算符 下表显示了 C 支持的算术运算符…

led灯性价比最高的是哪款?2023最新的哪种led灯质量比较好

Led作为一种固态半导体&#xff0c;具有自发光、高可控性的特点&#xff0c;高效节能&#xff0c;省电环保&#xff0c;并且根据材料的不同&#xff0c;发出的光线颜色也不同&#xff0c;因此广泛用于各种照明灯具&#xff0c;台灯也不例外。 作为一种适合工作学习或者工作休闲…

跑步耳机入耳好还是不入耳好、十大跑步运动耳机品牌排行榜推荐

健身房经常会播放一些节奏较快的歌曲&#xff0c;这样能够激发大家在运动过程中的动力&#xff0c;所以运动时聆听音乐确实比较有效果&#xff0c;居家运动、室外跑步时选择运动耳机就变成了刚需&#xff0c;首先不能影响其他人、佩戴时要稳定&#xff0c;音质和续航要有保证&a…

npm install 原理

npm基本使用node.jsNode.js 是一个基于 Chrome V8 引擎的 JavaScript 运行时环境。在任何地方安装了node&#xff0c;那么就可以运行JavaScript。所以在服务器上就可以运行js了。js就全栈了。node安装的同时就安装了npm。直接去官网下载&#xff0c;双击安装即可。安装完毕之后…

java基于springboot仓库管理系统的设计与实现

项目介绍 基于 SpringBoot 的仓库管理系统拥有多种角色&#xff0c;可供用户自定义使用。 功能模块 一、业务模块 1、客户管理 2、供应商管理 3、商品管理 4、商品进货管理 5、商品退货管理 6、商品销售管理 7、商品销售退货管理 二、系统模块 菜单管理&#xff0c;角色管…

Java高级工程师特别熟悉的js基础 | 面向对象编程/原型链/继承

面向对象编程/原型链/继承 面向对象编程 什么是面向对象编程&#xff1f; 面向对象是⼀种编程思想&#xff0c;经常被拿来和⾯面向过程⽐比较。其实说的简单点&#xff0c;面向过程关注的重点是动词&#xff0c;是分析出解决问题需要的步骤&#xff0c;然后编写函数实现每个步…

WIFI标签操作步骤

1. 打开并设置手机WIFI热点&#xff0c;设置SSID为ESL&#xff0c;密码为123456789&#xff08;如下图&#xff09; ​ 2. 运行APP&#xff0c;设置要接入的WIFI名称密码等信息&#xff08;如下图&#xff09; ​ 3. 长按背面按键&#xff08;长按2-3秒&#xff09;&#xff0c…

TypeScript 深度剖析:TypeScript 的理解?与 JavaScript 的区别?

一、是什么 TypeScript 是 JavaScript 的类型的超集&#xff0c;支持ES6语法&#xff0c;支持面向对象编程的概念&#xff0c;如类、接口、继承、泛型等 超集&#xff0c;不得不说另外一个概念&#xff0c;子集&#xff0c;怎么理解这两个呢&#xff0c;举个例子&#xff0c;如…

MYSQL 最大连接数

MYSQL 最大连接数&#xff1f; 最大连接数也就是mysql服务最大支持多少客户端连接服务 查看服务支持最大连接数&#xff1a; show variables like %max_connections%;如下: 如何查看 MYSQL 服务被多少个客户端连接了&#xff1f; show processlist;如下&#xff1a; 其中 …

数位dp--Windy数

题目 思路 大体思路其实也跟第一篇博客类似&#xff0c;用前缀和的思想处理区间的方案数&#xff0c;然后求方案数则是通过把每一位数都拆分来&#xff0c;然后根据两种选择0-an-1和选an两种情况进行判断&#xff0c;但是因为前导零会对结果产生不一样的结果&#xff08;比如0…

企业的ICP域名注册,你知道吗?

ICP域名备案查询的api接口&#xff0c;是企业在短时间内获得中沃公司最新数据的API接口。还可以帮助用户快速了解公司信息。基于域名的注册信息实时查询&#xff0c;适合对注册信息变更及时性要求较高的用户。通过ICP信息可以确认企业在互联网中的更多活动轨迹。企业的ICP域名注…

漏洞分析丨cve20110104

作者丨黑蛋目标程序调试工具16进制编辑器XP SP3office 2003ollydbg010Editor三、漏洞验证首先我们配置环境&#xff0c;并下载poc&#xff1a;使用ollydbg附加office excel 2003&#xff1a;打开poc可以看到发生了访问违规异常&#xff0c;像地址0x51453844中写入时发生异常&am…

【专项训练】分治、回溯

分治、回溯其实就是递归,只是是递归的一个细分,是一种特殊的递归 碰到一个题目,你就找他的重复性 最近重复性:根据重复性怎么构造以及如何分解,包括:分治、回溯 最优重复性:动态规划 本质:找重复性、分解问题、组合子问题的结果 回溯:试错! 50. Pow(x, n) https:…

sivalco使用测试pin正向导通

在学完功率半导体器件后&#xff0c;为了能够更好的了解功率半导体器件内部的相关性质&#xff0c;于是利用sivalco软件进行仿真&#xff0c;对于器件工作时相关数据进行分析&#xff0c;对于相关语句进行学习。 在功率半导体中以N或者P半导体进行掺杂获得我们所需要的器件&…