16.遍历二叉树,线索二叉树

news2024/12/23 17:27:54

目录

一. 遍历二叉树

(1)三种遍历方式

(2)递归遍历算法

(3)非递归遍历算法

(4)层次遍历算法

二. 基于递归遍历算法的二叉树有关算法

(1)二叉树的建立

(2)二叉树的复制

(3)二叉树的深度计算

(4)计算二叉树中的结点数

(5)计算二叉树中的叶子结点数

三. 线索二叉树


一. 遍历二叉树

遍历定义——顺着某一条搜索路径巡访二叉树中的结点,使得每个结点均被访问一次,而且仅被访问一次(又称周游)。这里“访问”的含义很广,可以是对结点作各种处理,如:输出结点的信息、修改结点的数据值等,但要求这种访问不破坏原来的数据结构。

遍历目的——得到树中所有结点的一个线性排列。

遍历用途——它是树结构插入、删除、修改、查找和排序运算的前提,是二叉树一切运算的基础和核心。

(1)三种遍历方式

由于二叉树有根结点D,左子树L,右子树R,只要依次访问这三部分,就可以遍历这个二叉树。规定先左后右,即必须先访问左子树,再访问右子树,则根据什么时候访问根结点有:DLR——先(根)序遍历,LDR———中(根)序遍历,LRD———后(根)序遍历三种遍历方式。

由二叉树的递归定义可知,遍历左子树和遍历右子树可如同遍历二叉树一样“递归”进行。

先序遍历的过程图解(1)
先序遍历的过程图解(2)
中序遍历的过程图解
后序遍历的过程图解

若二叉树中各结点的值均不相同,则二叉树结点的先序序列、中的序列和后序列都是唯一的。
下面我们考虑逆过程:由二叉树的先序序列和中序序列,或由二叉树的后序序列和中序序列可以确定唯一一棵二叉树(只知道前序和后序序列不可以)。

原理:先序(后序)序列的第一个(最后一个)一定是根结点,根结点在中序序列中把整个序列分为两部分,左边序列是左子树,右面序列是右子树。依此类推。

(2)递归遍历算法

下面以先序遍历为例写出递归算法代码。具体解释如下:

  1. 首先,判断二叉树是否为空(即 T==NULL),如果为空,则直接返回(return OK),因为空树没有节点可遍历。

  2. 如果二叉树不为空,首先对当前节点 T 进行访问操作(visit(T)),即对根节点进行处理。

  3. 然后,通过递归调用 PreOrderTraverse 函数来遍历当前节点的左子树(T->Ichild),即对左子树进行前序遍历。

  4. 接着,再通过递归调用 PreOrderTraverse 函数来遍历当前节点的右子树(T->rchild),即对右子树进行前序遍历。

这样,通过递归的方式,可以依次访问二叉树的根节点、左子树和右子树,实现了前序遍历。

需要注意的是,visit(T) 表示对节点 T 进行访问操作,具体的操作可以根据实际需求来定义。

void PreOrderTraverse(BiTree T){ //前序遍历
    if(T==NULL) 
        return OK; //空二叉树,这里OK理解为退出到上一层
    else{
        visit(T);//访问根结点
        PreOrderTraverse(T->Ichild);//递归遍历左子树
        PreOrderTraverse(T->rchild);//递归遍历右子树
    }
}

同样我们可以很容易的写出中序遍历和后序遍历的算法代码:

void PreOrderTraverse(BiTree T){ //中序遍历
    if(T==NULL) 
        return OK; //空二叉树
    else{
        PreOrderTraverse(T->Ichild);//递归遍历左子树
        visit(T);//访问根结点
        PreOrderTraverse(T->rchild);//递归遍历右子树
    }
}
void PreOrderTraverse(BiTree T){ //后序遍历
    if(T==NULL) 
        return OK; //空二叉树
    else{
        PreOrderTraverse(T->Ichild);//递归遍历左子树
        PreOrderTraverse(T->rchild);//递归遍历右子树
        visit(T);//访问根结点
    }
}

如果去掉输出语句,从递归的角度看,三种算法是完全相同的,或说这三种算法的访问路径是相同的,只是访问结点的时机不同。从图中虚线出发(白点代表左子树或者右子树为空),每个结点都经过了3次。第1次经过时访问=先序遍历,第2次经过时访问=中序遍历,第3次经过时访问=后序遍历。

遍历算法的时间复杂度:O(3n)=O(n),每个结点访问一次;

空间复杂度:O(n),对每一个经过但不访问的结点,我们都要找个空间存起来,最坏的情况就是除了根结点外其他所有结点都在右子树上,这时候存全部n个结点。

(3)非递归遍历算法

以中序遍历为例:二叉树中序遍历的非递归算法的关键:在中序遍历过某结点的整个左子树后,如何找到该结点的根以及右子树。
基本思想(中序遍历左子树根结点后进先出):

  • 建立一个栈
  • 根结点进栈,遍历左子树
  • 左子树全部访问完毕,根结点出栈,输出根结点,遍历右子树。

栈的变化情况:A进-B进-B出-D进-D出-A出-C进-C出;

写出代码,具体步骤解释如下:

  1. 首先,声明一个辅助指针 p 和一个栈 S,用于存储节点和辅助遍历。

  2. 初始化栈 S,即将栈置空。

  3. 将根节点 T 赋值给指针 p,即将 p 指向根节点。

  4. 进入循环,判断条件为 p 非空或者栈 S 非空。只要满足这个条件,就继续遍历二叉树。

  5. 在循环中,首先判断当前节点 p 是否非空。如果非空,则将其入栈(Push(S,p)),并将 p 指向其左子树(p = p->lchild)。

  6. 如果当前节点 p 为空,即左子树为空,说明已经到达最左边的叶子节点。此时需要从栈中弹出一个节点(Pop(S,q))。

  7. 对弹出的节点 q 进行访问操作,这里使用 printf 打印出节点的数据(printf("%c",q->data))。

  8. 将指针 p 指向节点 q 的右子树(p = q->rchild),继续遍历右子树。

  9. 重复步骤 4-8,直到遍历完整个二叉树。

  10. 最后,当指针变量p指向NULL,且工作栈中没有元素,说明整个二叉树遍历结束,跳出整个while循环,返回状态值 OK,表示函数执行成功。

Status InOrderTraverse(BiTree T){
    BiTree p,q;  //弄两个指针 
    InitStack(S);  //初始化一个工作栈
    p=T;  //让p指向树的根结点
    while(p||!StackEmpty(S)){  //StackEmpty()栈为空返回TRUE,否则返回FALSE
        if(p){
            Push(S,p);
            p = p->lchild;
        }
        else{
            Pop(S,q); 
            printf(“%c”,q->data);
            p = q->rchild;
        }
    }//while
return OK;
}

(4)层次遍历算法

二叉树的层次遍历:对于一颗二叉树,从根结点开始,按从上到下、从左到右的顺序访问每一个结点。每一个结点仅仅访问一次。

算法设计思路:使用队列。
I. 将根结点进队;
II. 队不空时循环:从队列中出列一个结点*p,访问它,依次执行下面两步:

  • 若它有左孩子结点,将左孩子结点进队;
  • 若它有右孩子结点,将右孩子结点进队。

队列变化过程:a进队-a出队,b,f依次进队-b出队,c,d依次进队-f出队,g进队-c出队-d出队,e进队-g出队,h进队-e出队-h出队

使用队列定义类型如下:

typedef struct{
    BTNode data[MaxSize];  //存放队中元素
    int front,rear;  //队头和队尾指针
}SqQueue;  //顺序循环队列类型

给出算法代码,解释如下:

  1. 首先,声明一个指针 p 和一个队列 qu,用于存储节点和辅助遍历。

  2. 初始化队列 qu,即将队列置空。

  3. 将根结点 b 入队列(enQueue(qu,b)),即将根节点放入队列中。

  4. 进入循环,判断条件为队列 qu 非空。只要队列非空,就继续遍历二叉树。

  5. 在循环中,首先从队列中出队列一个节点(deQueue(qu,p)),并对该节点进行访问操作,这里使用 printf 打印出节点的数据(printf(" %c", p->data))。

  6. 然后,判断当前节点 p 的左子节点是否非空。如果非空,则将其入队列(enQueue(qu,p->lchild))。

  7. 接着,判断当前节点 p 的右子节点是否非空。如果非空,则将其入队列(enQueue(qu,p->rchild))。

  8. 重复步骤 4-7,直到遍历完整个二叉树。

  9. 最后,返回结果。

void LevelOrder(BTNode *b) {
    BTNode *p;  //p指向队列头部用于出队列
    SqQueue qu;
    InitQueue(qu);  //初始化队列
    enQueue(qu,b);  //根结点指针进入队列
    while(!QueueEmpty(qu)){ //队不为空,则循环
        deQueue(qu,p);  //出队结点p
        printf(" %c", p->data);  //访问结点p
        if (p->Ichild!=NULL) enQueue(qu,p->lchild);  //有左孩子时将其进队
        if (p->rchild!=NULL) enQueue(qu,p->rchild);  //有右孩子时将其进队
    }  
}

二. 基于递归遍历算法的二叉树有关算法

(1)二叉树的建立

(以先序遍历为例)分两步进行:(1)从键盘输入二叉树的结点信息,建立二叉树的存储结构;(2)在建立二叉树的过程中按照二叉树先序方式建立;

先序遍历:ABCDEGF,有不同的树,所以加上空结点,就有唯一的树结构
Status CreateBiTree(BiTree &T){  //建立二叉树,传入指针类型
    scanf(&ch); //从键盘输入,C++中cin>>ch;
    if(ch == “#”) T=NULL;
    else{
        if(!(T=(BiTNode*)malloc(sizeof(BiTNode))))
            exit(OVERFLOW); //C++中T=new BiTNode;
        T->data = ch;  //生成根结点
        CreateBiTree(T->Ichild);  //构造左子树
        CreateBiTree(T->rchild);  //构造右子树
    }
    return OK;  //同理,这里return到上一层调用
}  //CreateBiTree

这段代码是用来创建二叉树的函数。函数的输入参数是一个指向二叉树根节点的指针T。首先通过scanf函数读取一个字符ch作为输入。如果ch等于"#",表示当前节点为空,将T指向NULL。否则,进入else语句块。在else语句块中,首先通过malloc函数为当前节点分配内存空间,并将分配的地址赋值给T。如果分配内存失败,则程序退出。然后将当前节点的数据域赋值为ch。接下来递归调用CreateBiTree函数,创建当前节点的左子树,将左子树的根节点地址赋值给T->lchild。然后再次递归调用CreateBiTree函数,创建当前节点的右子树,将右子树的根节点地址赋值给T->rchild。最后,函数返回OK表示创建二叉树成功。

利用上面的代码,输入ABC##DE#G##F###,就可建立下面的二叉树链表。

(2)二叉树的复制

int Copy(BiTree T,BiTree &NewT){
    if(T == NULL){  //如果是空树返回0
        NewT = NULL;
        return 0;
    }
    else{
        NewT = new BiTNode;  //建立新结点,NewT指针指向它
        NewT->data = T->data;  //传入数据
        Copy(T->lChild,NewT->lchild);  //递归调用,复制左子树
        Copy(T->rChild,NewT->rchild);  //递归调用,复制右子树
    }
}

这段代码是一个递归函数,用于复制二叉树。函数的参数是原始二叉树T和目标二叉树NewT(通过引用传递),返回值为0表示复制成功。需要注意的是,在函数中使用了引用传递(&)来传递目标二叉树NewT的地址,这是为了能够在函数内部修改目标二叉树NewT的指针,以便将复制得到的子树链接到正确的位置上。

(3)二叉树的深度计算

int Depth(BiTree T){
    if(T==NULL) return 0;  //如果是空树返回0
    else{
        m = Depth(T->lChild);
        n = Depth(T->rChild);
        if(m>n) return (m+1);
        else return(n+1);
    }
}

这段代码是一个递归函数,用于计算二叉树的深度。函数的参数是二叉树T,返回值为二叉树的深度。首先,函数会检查二叉树T是否为空(即是否为叶子节点)。如果是空树,即T为NULL,那么二叉树的深度为0,函数直接返回0。

如果二叉树T不为空,那么函数会递归调用自身,分别计算左子树和右子树的深度,并将结果分别赋值给变量m和n。接下来,函数会比较左子树的深度m和右子树的深度n的大小。如果m大于n,说明左子树更深,那么函数返回m+1,表示整个二叉树的深度为左子树的深度加1。如果n大于等于m,说明右子树更深或者左右子树深度相等,那么函数返回n+1,表示整个二叉树的深度为右子树的深度加1。

最终,当递归调用完成后,函数会返回整个二叉树的深度。

(4)计算二叉树中的结点数

int NodeCount(BiTree T){
    if(T == NULL)
        return 0;
    else
        return NodeCount(T->lchild)+NodeCount(T->rchild)+1;
}

如果是空树,即T为NULL,那么二叉树中节点的数量为0,函数直接返回0。

如果二叉树T不为空,结点个数=左子树结点数量+右子树结点数量+当前节点(即加1)。

(5)计算二叉树中的叶子结点数

int LeadCount(BiTree T){
    if(T==NULL)  //如果是空树返回0
        return 0;
    if (T->Ichild == NULL && T->rchild == NULL)
        return 1;  //如果是叶子结点返回1
    else
        return LeafCount(T->lchild) + LeafCount(T->rchild);
}

如果是空树,则叶子结点个数为0;
否则,为左子树的叶子结点个数+右子树的叶子结点个数。

三. 线索二叉树

当用二叉链表作为二叉树的存储结构时,可以很方便地找到某个结点的左右孩子;但一般情况下,无法直接找到该结点在某种遍历序列中的前驱和后继结点。为了解决这个问题,一般有以下解决方案:

  • 通过遍历寻找——费时间
  • 再增设前驱、后继指针域——增加了存储负担。
  • 利用二叉链表中的空指针域。

这里我们重点介绍最后一种。首先回忆下二叉树链表中空指针域的数量:具有n个结点的二叉链表中,一共有2n个指针域(每个结点有左右指针域两个);因为n个结点中有n-1个孩子,即2n个指针域中,有n-1个用来指示结点的左右孩子,其余n+1个指针域为空。

如果某个结点的左孩子为空,则将空的左孩子指针域改为指向其前驱;如果某结点的右孩子为空,则将空的右孩子指针域改为指向其后继。这种改变指向的指针称为“线索",加上了线索的二叉树称为线索二叉树(Threaded Binary Tree)。

为区分lrchid和rchild指针到底是指向孩子的指针,还是指向前驱现白后继的指针,对二叉链表中每个结点增设两个标志域Itag 和rtag,并约定:

  • ltag = 0; lchild指向该结点的左孩子;
  • ltag = 1; lchild指向该结点的前驱;
  • rtag = 0; rchild指向该结点的右孩子;
  • rtag = 1; rchild指向该结点的后继;

这样,结点的结构就由5部分组成:

typedef struct BiThrNode{
    int data;
    int Itag, rtag;
    struct BiThrNode *Ichild,*rchild;
}BiThrNode,*BiThrTree;
先序遍历的线索二叉树
后序遍历的线索二叉树

为避免有些指针处于悬空状态,增设了一个头结点,这个头结点还是BiThrNode类型:
ltag=0, lchild指向根结点;
rtag=1, rchild指向遍历序列中最后一个结点;
遍历序列中第一个结点的lc域和最后一个结点的rc域都指向头结点;

这样,改进完的线索二叉树如下:

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

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

相关文章

小程序中的页面配置和网络数据请求

页面配置文件和常用的配置项 1.在msg.json中配置window中的颜色和背景色 "navigationBarBackgroundColor": "#efefef","navigationBarTextStyle": "black" 2.可以看到home中的没有发生变化但是msg的发生变化了,这个和前面的…

Mysql查询重复数据常用方法

在平常的开发工作中,我们经常需要查询数据,比如查询某个表中重复的数据,那么,具体应该怎么实现呢?常用的方法都有哪些呢? 测试表中数据: 1:查询名字重复的数据 having: …

面试之快速学习计算机网络-http

1. HTTP常见状态码 2. 3开头重定向,4开头客户端错误,5开头服务端错误 2. HTTP 报文 1. start-line:请求行,可以为以下两者之一: 请求行: GET /hello-world2.html HTTP/1.1状态行:HTTP/1.1 200…

关于模板的大致认识【C++】

文章目录 函数模板函数模板的原理函数模板的实例化模板参数的匹配原则 类模板类模板的定义格式类模板的实例化 非类型模板参数typename 与class模板的特化函数模板特化类模板特化全特化偏特化 模板的分离编译 函数模板 函数模板的原理 template <typename T> //模板参数…

el-input输入框 输入数字中文 来回切换之后 监听失效问题如何解决

实现一个vue自定义指令——输入框&#xff08;input,el-input&#xff09;输入内容类型限制&#xff0c;解决中文输入法双向绑定失效问题&#xff0c;多种类型支持&#xff0c;数字类型&#xff0c;浮点类型、英文类型、整数类型、四则运算等 直接上代码 首先新建input.js ex…

Git如何操作本地分支仓库?

基本使用TortoiseGit 操作本地仓库(分支) 分支的概念 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着你可以把你的工作从开发主线上分离开来&#xff0c;避免影响开发主线。多线程开发,可以同时开启多个任务的开发&#xff0c;多个任务之间互不影响。 为何要…

10个好用的网络画图工具推荐,专业办公绘图必备!

在当今数字化时代&#xff0c;网络画图工具成为了各行各业的重要辅助工具。无论是制作流程图、思维导图、原型设计&#xff0c;还是插图绘制、数据可视化&#xff0c;网络画图工具为用户提供了便捷、高效的创作平台。本文将向大家推荐10个好用的网络画图工具&#xff0c;帮助你…

深度解析淘宝API商品评论接口的实现原理与使用方法

淘宝API商品评论接口&#xff0c;主要用于获取某个商品的评价信息。通过该接口&#xff0c;我们可以获取到商品的所有评价内容、评价时间、评价等级等相关信息&#xff0c;帮助我们更好地了解用户对商品的反馈&#xff0c;进而进行数据分析和业务优化。 一、接口鉴权 在使用淘…

波奇学C++:stl的list模拟实现

list是双向带头链表。所以迭代器end()相当于哨兵卫的头。 list不支持和[]重载&#xff0c;原因在于list空间不是连续的&#xff0c;和[]的代价比较大。 访问第n个节点&#xff0c;只能用for循环&#xff0c;来实现 list<int> l; l.push_back(0); l.push_back(1); l.pu…

代码随想录算法训练营之JAVA|第三十五天|343. 整数拆分

今天是第 天刷leetcode&#xff0c;立个flag&#xff0c;打卡60天&#xff0c;如果做不到&#xff0c;完成一件评论区点赞最高的挑战。 算法挑战链接 343. 整数拆分https://leetcode.cn/problems/integer-break/ 第一想法 题目理解&#xff1a;将一个整数拆分为k个整数&…

python matlab 画坐标图

画一个坐标系&#xff0c;同时显示两条直线&#xff0c;效果图如下&#xff1a; 功能点&#xff1a; 同时显示两个纵坐标数据 显示图片名称 图片最大化保存 到本地 在图片某个位置显示字符信息 不同的线名称提示 代码如下&#xff1a; import matplotlib.pyplot as pltde…

学习左耳听风栏目90天——第七天 7/90(学习左耳朵耗子的工匠精神,对技术的热爱)【每个程序员都该知道的事】

每个程序员都该知道的事 每个程序员都应该要读的书每个搞计算机专业的学生应有的知识LinkedIn 高效的代码复查技巧编程语言和代码质量的研究报告 每个程序员都应该要读的书 每个搞计算机专业的学生应有的知识 LinkedIn 高效的代码复查技巧 编程语言和代码质量的研究报告

C++头文件和std命名空间

C 是在C语言的基础上开发的&#xff0c;早期的 C 还不完善&#xff0c;不支持命名空间&#xff0c;没有自己的编译器&#xff0c;而是将 C 代码翻译成C代码&#xff0c;再通过C编译器完成编译。 这个时候的 C 仍然在使用C语言的库&#xff0c;stdio.h、stdlib.h、string.h 等头…

CentOS7网络配置

本文是我从另外三个文章中整合而来&#xff0c;用于自存&#xff0c;如有侵权请联系我删除。 CentOS 7教程&#xff08;二&#xff09;-网络设置 - 知乎 (zhihu.com) VMware安装、Linux下CentOS7的配置及网络环境的配置&#xff08;最新版特别全&#xff09;_centos7 配置_Co…

田间气象站的优势与应用

在农业生产中&#xff0c;田间气象站是重要的气象监测工具&#xff0c;它能够对农田间的气象信息进行实时监测和记录&#xff0c;为农民伯伯提供农业生产科学依据。 田间气象站是由多个传感器共同组成&#xff0c;能够收集各项气象参数&#xff0c;包括我们常见的风速、风向、…

STM32--MPU6050与I2C外设

文章目录 前言MPU6050参数电路MPU6050框图 IIC外设框图 IIC的基本结构软件IIC实现MPU6050硬件IIC实现MPU6050 前言 在51单片机专栏中&#xff0c;用过I2C通信来进行实现AT24C02的数据存储&#xff1b; 里面介绍的是利用程序的编程来实现I2C的时序&#xff0c;进而实现AT24C02与…

关于android studio 几个简单的问题说明

自信是成功的第一步。——爱迪生 1. android studio 如何运行不同项目是否要更换不同的sdk 和 gradle 2.编译Gradle总是错误为什么 3.如何清理android studio 的缓存 4. 关于android Studio中的build 下面的rebuild project

『SEQ日志』在 .NET Core 中轻松集成 NLog 和 SEQ ,打造轻量级的分布式日志系统

&#x1f4e3;读完这篇文章里你能收获到 如何在Docker中部署 SEQ&#xff1a;介绍了如何创建和运行 SEQ 容器&#xff0c;给出了详细的执行操作如何使用 NLog 接入 .NET Core 应用程序的日志&#xff1a;详细介绍了 NLog 和 NLog.Seq 来配置和记录日志的步骤日志记录示例&…

判断三角形

int main() {int a 0;int b 0;int c 0;scanf("%d%d%d", &a, &b, &c);if ((ab>c)&&(ac>b)&&(bc>a)){if (a b && b c){printf("等边三角形\n");}else if ((a b && b ! c) || (a c && c…

企业的降本增益,从源头抓起——精细化管理每一条销售线索

存量经济时代&#xff0c;降本增效成为企业增长的重点抓手&#xff0c;也是企业自身数字化建设重要目标。 如何能让销售更珍惜线索&#xff1f;如何能让线索产生更快的转化&#xff1f;如何能让对的线索给到对的销售&#xff1f;如何能依托线索转化结果找到更有效的获客渠道&a…