数据结构 -- 线索二叉树

news2025/3/25 11:59:42

线索二叉树

线索二叉树的概念

线索二叉树的作用

我们在进行中序遍历时,总是从根节点出发进行二叉树遍历,而当仅知道某一孩子节点的指针时,由于无法访问父节点,所以没有办法进行遍历。由此引入线索二叉树

【思考】①如何找到指定结点p在中序遍历序列中的前驱?②如何找到p的后继

【思路】从根节点出发,重新进行依次中序遍历,指针q记录当前访问的结点,指针pre记录上一个被访问的结点①当q == p时,pre为前驱②当pre == p时,q为后继

【缺点】很不方便,遍历操作必须从根结点开始

【解决】引入线索二叉树

n个节点的二叉树,由n+1个空链域!可以用来记录前驱、后继的信息

前驱线索由左孩子充当,后驱线索由右孩子充当(如果没有前驱或没有后继则左/右孩子为NULL)

线索二叉树的存储结构

增加左右线索标志 ltag 、rtag

tag == 0,表示指针指向孩子

tag == 1,表示指针是“线索”

//线索二叉树的结点(链式存储)
typedef struct ThreadNode{
    ElemType data;						//数据域
    struct ThreadNode * lchild,*rchild;	//左右孩子
    int ltag,rtag;			//左右线索
}ThreadNode,*ThreadTree;
三种线索二叉树

中序线索二叉树:以中序遍历序列为依据进行“线索化”

先序线索二叉树:以先序遍历序列为依据进行“线索化”

后序线索二叉树:以后序遍历序列为依据进行“线索化”

几个概念:

线索:指向前驱/后继的指针称为线索

中序前驱/中序后继;先序前驱/先序后继;后序前驱/后序后继

手算画出线索二叉树

① 确定线索二叉树类型(中序、先序、后序)

② 按照对应遍历规则,确定各个结点的访问顺序,并写上编号

③ 将n+1个空链域脸上前驱、后继

二叉树的线索化

中序线索化
//线索二叉树的结点(链式存储)
typedef struct ThreadNode{
    ElemType data;						//数据域
    struct ThreadNode * lchild,*rchild;	//左右孩子
    int ltag,rtag;			//左右线索
}ThreadNode,*ThreadTree;

//全局变量pre,指向当前访问的结点的前驱
ThreadNode *pre = NULL;

//中序线索化二叉树T
void CreateInThread(TreadTree T){
    pre = NULL;					//pre初始化为NULL
    if(T != NULL){				//非空二叉树才能线索化
        InThread(T);			//中序线索化二叉树
        if(pre->rchild == NULL)	pre->rtag = 1;	//处理遍历后的最后一个结点
    }
}

//中序遍历二叉树,一边遍历一遍线索化
void InThread(Thread T){
    if(T != NULL){
        InThread(T->lchild);	//中序遍历左子树
        visit(T);				//访问根结点
        InThread(T->rchild);	//中序遍历右子树
    }
}

void visit(ThreadNode *q){
    if(q->lchild == NULL){//左子树为空,建立前驱线索
        q->lchild = pre;
        q->ltag = 1;
    }
    if(pre != NULL && pre->rchild == NULL){
        pre->rchild = q;	//建立前驱结点的后继线索
        pre->rtag = 1;
    }
    pre = q;
}
先序线索化
//线索二叉树的结点(链式存储)
typedef struct ThreadNode{
    ElemType data;						//数据域
    struct ThreadNode * lchild,*rchild;	//左右孩子
    int ltag,rtag;			//左右线索
}ThreadNode,*ThreadTree;

//全局变量pre,指向当前访问的结点的前驱
ThreadNode *pre = NULL;

//先序线索化二叉树T
void CreatePreThread(TreadTree T){
    pre = NULL;					//pre初始化为NULL
    if(T != NULL){				//非空二叉树才能线索化
        PreThread(T);			//先序线索化二叉树
        if(pre->rchild == NULL)	pre->rtag = 1;	//处理遍历后的最后一个结点
    }
}

//先序遍历二叉树,一边遍历一遍线索化
void PreThread(ThreadTree T){
    if(T != NULL){
        visit(T);				//访问根结点
        if(T->ltag == 0)	PreThread(T->lchild);	//先序遍历左子树
        PreThread(T->rchild);	//先序遍历右子树
    }
}

void visit(ThreadNode *q){
    if(q->lchild == NULL){//左子树为空,建立前驱线索
        q->lchild = pre;
        q->ltag = 1;
    }
    if(pre != NULL && pre->rchild == NULL){
        pre->rchild = q;	//建立前驱结点的后继线索
        pre->rtag = 1;
    }
    pre = q;
}

【注意】在线序线索化过程中,在访问完左孩子之后会返回根结点导致循环,所以要在遍历左子树之前先判断当前的左孩子是真正的左孩子还是线索

后序线索化
//线索二叉树的结点(链式存储)
typedef struct ThreadNode{
    ElemType data;						//数据域
    struct ThreadNode * lchild,*rchild;	//左右孩子
    int ltag,rtag;			//左右线索
}ThreadNode,*ThreadTree;

//全局变量pre,指向当前访问的结点的前驱
ThreadNode *pre = NULL;

//后序线索化二叉树T
void CreatePostThread(TreadTree T){
    pre = NULL;					//pre初始化为NULL
    if(T != NULL){				//非空二叉树才能线索化
        PostThread(T);			//后序线索化二叉树
        if(pre->rchild == NULL)	pre->rtag = 1;	//处理遍历后的最后一个结点
    }
}

//后序遍历二叉树,一边遍历一遍线索化
void PostThread(ThreadTree T){
    if(T != NULL){
        PostThread(T->lchild);	//后序遍历左子树
        PostThread(T->rchild);	//后序遍历右子树
        visit(T);				//访问根结点
    }
}

void visit(ThreadNode *q){
    if(q->lchild == NULL){//左子树为空,建立前驱线索
        q->lchild = pre;
        q->ltag = 1;
    }
    if(pre != NULL && pre->rchild == NULL){
        pre->rchild = q;	//建立前驱结点的后继线索
        pre->rtag = 1;
    }
    pre = q;
}

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

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

相关文章

【算法day19】括号生成——数字 n 代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且 有效的 括号组合。

括号生成 https://leetcode.cn/problems/generate-parentheses/description/ 数字 n 代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且 有效的 括号组合。 左括号数必须大于右括号数,且小于等于n class Solution { publ…

Qt5.15.2实现Qt for WebAssembly与示例

目录 1.什么是Qt for WebAssembly? 1.1 什么是 WebAssembly? 1.2 WebAssembly 的优势 1.3 什么是 Qt for WebAssembly? 1.4 Qt for WebAssembly 的特点 1.5 编译过程 1.6 运行时环境 注意!!!注意&am…

好吧好吧,看一下达梦的模式与用户的关系

单凭个人感觉,模式在达梦中属于逻辑对象合集,回头再看资料 应该是一个用户可以对应多个模式 问题来了,模式的ID和用户的ID一样吗? 不一样 SELECT USER_ID,USERNAME FROM DBA_USERS WHERE USERNAMETEST1; SELECT ID AS SCHID, NA…

HOW - DP 动态规划系列(三)(含01背包问题)

目录 一、01背包问题最直接的暴力解法动态规划解法 二、完全背包 通过几个算法的学习,理解和掌握动态规划来解决背包问题。 一、01背包问题 对于面试的话,掌握01背包和完全背包就够用了,最多可以再来一个多重背包。 如果这几种背包分不清&…

在linux服务器部署Heygem

前言: Heygem官方文档上提供了基于windwos系统的安装方案。在实际使用过程中个人电脑的配置可能不够。这个时候如果服务器配置够的话,可以尝试在服务器上装一下。但是服务器一般都是linux系统的,于是这篇教程就出现了… 可行性分析 通读安装…

图书管理系统系统-Java、SpringBoot、Vue和MySQL开发的图书馆管理系统

「springboot、vue图书馆管理系统.zip」 链接:https://pan.quark.cn/s/5a929a7e9450 分享一个图书管理系统,Java、SpringBoot、Vue和MySQL开发的图书馆管理系统 以下是对文本内容的总结: 项目概述 项目名称与背景: 项目概述 项…

[c语言日寄]数据输入

【作者主页】siy2333 【专栏介绍】⌈c语言日寄⌋:这是一个专注于C语言刷题的专栏,精选题目,搭配详细题解、拓展算法。从基础语法到复杂算法,题目涉及的知识点全面覆盖,助力你系统提升。无论你是初学者,还是…

字节DAPO算法:改进DeepSeek的GRPO算法-解锁大规模LLM强化学习的新篇章(代码实现)

DAPO算法:解锁大规模LLM强化学习的新篇章 近年来,大规模语言模型(LLM)在推理任务上的表现令人瞩目,尤其是在数学竞赛(如AIME)和编程任务中,强化学习(RL)成为…

计算机操作系统(四) 操作系统的结构与系统调用

计算机操作系统(四) 操作系统的结构与系统调用 前言一、操作系统的结构1.1 简单结构1.2 模块化结构1.3 分层化结构1.4 微内核结构1.5 外核结构 二、系统调用1.1 系统调用的基本概念1.2 系统调用的类型 总结(核心概念速记)&#xf…

DeepSeek技术架构解析:MoE混合专家模型

一、前言 2025年初,DeepSeek V3以557万美元的研发成本(仅为GPT-4的1/14)和开源模型第一的排名,在全球AI领域掀起波澜。其核心创新之一——混合专家模型(Mixture of Experts, MoE)的优化设计,不…

【正点原子】AI人工智能深度学习(RV1126/RK3568/RK3588)-第1期 准备篇

1.1SDK编译后的目录 1、真正的根文件系统镜像存放目录 2、非必须,负责系统升级等,kerneldtbramdisk组成的根文件系统 1.2文件系统分区 1.3开机自启动 1.6设置静态ip地址 1.8RKMedia框架/编译测试SDK自带RKMedia例程 出厂系统以下内容都是默认…

靶场(十五)---小白心得思路分析---LaVita

启程: 扫描端口,发现开放22,80端口,发现ws.css可能存在exp,经查发现无可利用的exp PORT STATE SERVICE VERSION 22/tcp open ssh OpenSSH 8.4p1 Debian 5deb11u2 (protocol 2.0) | ssh-hostkey: | 3072 c9…

【AI大模型】DeepSeek + 通义万相高效制作AI视频实战详解

目录 一、前言 二、AI视频概述 2.1 什么是AI视频 2.2 AI视频核心特点 2.3 AI视频应用场景 三、通义万相介绍 3.1 通义万相概述 3.1.1 什么是通义万相 3.2 通义万相核心特点 3.3 通义万相技术特点 3.4 通义万相应用场景 四、DeepSeek 通义万相制作AI视频流程 4.1 D…

Pi型隶属函数(Π-shaped Membership Function)的详细介绍及python示例

我们前文已经深度解读了三角形、梯形、高斯、S型和Z型隶属函数,现在转向Pi型。当然我们先简要回顾不同隶属函数的特点和曲线效果。了解每种隶属函数的特性是为了更好的应用。 一、回顾五种隶属函数的特点 1.从每种隶属函数的结构和特点角度对比。三角形隶属函数&am…

MySQL 入门大全:常用函数

🧑 博主简介:CSDN博客专家,历代文学网(PC端可以访问:https://literature.sinhy.com/#/literature?__c1000,移动端可微信小程序搜索“历代文学”)总架构师,15年工作经验,…

SpringBoot3实战(SpringBoot3+Vue3基本增删改查、前后端通信交互、配置后端跨域请求、数据批量删除(超详细))(3)

目录 一、从0快速搭建SpringBoot3工程、SpringBoot3集成MyBatis、PageHelper分页查询的详细教程。(博客链接) 二、实现前端与后端通信对接数据。(axios工具) &#xff08;1&#xff09;安装axios。(vue工程目录) &#xff08;2&#xff09;封装请求工具类。(request.js) <1&…

JVM垃圾回收笔记02-垃圾回收器

文章目录 前言1.串行(Serial 收集器/Serial Old 收集器)Serial 收集器Serial Old 收集器相关参数-XX:UseSerialGC 2.吞吐量优先(Parallel Scavenge 收集器/Parallel Old 收集器)Parallel Scavenge 收集器Parallel Old 收集器相关参数-XX:UseParallelGC ~ -XX:UseParallelOldGC-…

Linux上位机开发实战(编写API库)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 我们自己编写linux上位机软件的时候&#xff0c;尽量都是通过框架库的形式来开发。这就是所谓的低耦合&#xff0c;高内聚。相似的功能、模块和算法…

器件功耗模型原理

器件功耗模型原理 谷歌提供了一套通用的器件耗电模型和配置方案&#xff0c;先对器件进行耗电因子拆解&#xff0c;建立器件功耗模型&#xff0c;得到一个器件耗电的计算公式。通过运行时统计器件的使用数据&#xff0c;代入功耗模型&#xff0c;就可以计算出器件的功耗。例如…

拥抱成长型思维:解锁持续进步的人生密码

我强烈推荐4本可以改变命运的经典著作&#xff1a; 《寿康宝鉴》在线阅读白话文《欲海回狂》在线阅读白话文《阴律无情》在线阅读白话文《了凡四训》在线阅读白话文 一、什么是成长型思维&#xff1f; 成长型思维&#xff08;Growth Mindset&#xff09;由斯坦福大学心理学家卡…