【数据结构】树与二叉树(廿一):树和森林的遍历——先根遍历(递归算法PreOrder、非递归算法NPO)

文章目录

  • 5.1 树的基本概念
    • 5.1.1 树的定义
    • 5.1.2 森林的定义
    • 5.1.3 树的术语
  • 5.2 二叉树
  • 5.3 树
    • 5.3.1 树的存储结构
      • 1. 理论基础
      • 2. 典型实例
      • 3. Father链接结构
      • 4. 儿子链表链接结构
      • 5. 左儿子右兄弟链接结构
    • 5.3.2 获取结点的算法
    • 5.3.3 树和森林的遍历
      • 1. 先根遍历(递归)
        • a.理论
        • b. ADL算法PreOrder
        • c. 代码实现
      • 2. 先根遍历(非递归)
        • a. ADL算法NPO
        • b. NPO算法解析
        • c. 代码实现
      • 3. 代码整合

5.1 树的基本概念

5.1.1 树的定义

  • 一棵树是结点的有限集合T:
    • 若T非空,则:
      • 有一个特别标出的结点,称作该树的,记为root(T);
      • 其余结点分成若干个不相交的非空集合T1, T2, …, Tm (m>0),其中T1, T2, …, Tm又都是树,称作root(T)的子树
    • T 空时为空树,记作root(T)=NULL。

5.1.2 森林的定义

  一个森林是0棵或多棵不相交(非空)树的集合,通常是一个有序的集合。换句话说,森林由多个树组成,这些树之间没有交集,且可以按照一定的次序排列。在森林中,每棵树都是独立的,具有根节点和子树,树与树之间没有直接的连接关系。
  森林是树的扩展概念,它是由多个树组成的集合。在计算机科学中,森林也被广泛应用于数据结构和算法设计中,特别是在图论和网络分析等领域。
在这里插入图片描述

5.1.3 树的术语

  • 父亲(parent)、儿子(child)、兄弟(sibling)、后裔(descendant)、祖先(ancestor)
  • 度(degree)、叶子节点(leaf node)、分支节点(internal node)
  • 结点的层数
  • 路径、路径长度、结点的深度、树的深度

参照前文:【数据结构】树与二叉树(一):树(森林)的基本概念:父亲、儿子、兄弟、后裔、祖先、度、叶子结点、分支结点、结点的层数、路径、路径长度、结点的深度、树的深度

5.2 二叉树

5.3 树

5.3.1 树的存储结构

1. 理论基础

2. 典型实例

3. Father链接结构

4. 儿子链表链接结构

【数据结构】树与二叉树(十八):树的存储结构——Father链接结构、儿子链表链接结构

5. 左儿子右兄弟链接结构

【数据结构】树与二叉树(十九):树的存储结构——左儿子右兄弟链接结构(树、森林与二叉树的转化)
  左儿子右兄弟链接结构通过使用每个节点的三个域(FirstChild、Data、NextBrother)来构建一棵树,同时使得树具有二叉树的性质。具体来说,每个节点包含以下信息:

  1. FirstChild: 存放指向该节点的大儿子(最左边的子节点)的指针。这个指针使得我们可以迅速找到一个节点的第一个子节点。
  2. Data: 存放节点的数据。
  3. NextBrother: 存放指向该节点的大兄弟(同一层中右边的兄弟节点)的指针。这个指针使得我们可以在同一层中迅速找到节点的下一个兄弟节点。

  通过这样的结构,整棵树可以用左儿子右兄弟链接结构表示成一棵二叉树。这种表示方式有时候被用于一些特殊的树结构,例如二叉树、二叉树的森林等。这种结构的优点之一是它更紧凑地表示树,而不需要额外的指针来表示兄弟关系。
在这里插入图片描述

   A/|\B C D/ \E   F
A
|
B -- C -- D|E -- F

即:

      A/ B   \C/ \ E   D\F

在这里插入图片描述

5.3.2 获取结点的算法

【数据结构】树与二叉树(二十):树获取大儿子、大兄弟结点的算法(GFC、GNB)

5.3.3 树和森林的遍历

1. 先根遍历(递归)

【数据结构】树与二叉树(七):二叉树的遍历(先序、中序、后序及其C语言实现)

a.理论

在这里插入图片描述

b. ADL算法PreOrder

在这里插入图片描述

  1. 基本条件检查:

    • IF t=NULL THEN RETURN.:如果树的根节点 t 为空,直接返回,递归的出口条件。
  2. 打印根节点数据:

    • PRINT(Data(t)).:打印当前树节点 t 的数据。
  3. 递归调用子树的先根遍历:

    • PreOrder(t.child).:递归调用先根遍历算法,对当前节点 t 的第一个孩子进行遍历。
  4. 迭代调用右兄弟节点的先根遍历:

    • WHILE child≠∧ DO:使用 WHILE 循环,判断当前节点的第一个孩子是否存在(child≠∧)。
      • PreOrder(child).:递归调用先根遍历算法,对当前节点 child 进行遍历。
      • GNB(child.child).:调用算法 GNB 获取当前节点 child 的下一个兄弟节点,然后继续遍历。

  通过递归地调用先根遍历算法,依次访问树的根节点、根节点的孩子节点、孩子节点的兄弟节点,以此类推,完成对整个树的先根遍历。

c. 代码实现
void PreOrder(TreeNode* t) {// 基本条件检查if (t == NULL) {return;}// 打印当前树节点的数据printf("%c ", t->data);// 递归调用子树的先根遍历TreeNode* child = getFirstChild(t);while (child != NULL) {PreOrder(child);// 迭代调用右兄弟节点的先根遍历child = getNextBrother(child);}
}

2. 先根遍历(非递归)

a. ADL算法NPO

在这里插入图片描述

b. NPO算法解析
  1. 栈的初始化:

    • CREATE(S): 创建一个栈 S 用于存储待访问的节点。
  2. 初始节点指针 p 的设置:

    • p ← t: 将当前节点指针 p 设置为树的根节点 t
  3. 遍历过程:

    • NPO3. [若 p 所指结点不为空,访问 p 所指结点,将 p 压入栈,并将其 FirstChild 指针设为 p.]
      • 如果当前节点 p 不为空,访问该节点的数据,将 p 压入栈,并将 p 的第一个孩子节点设置为新的 p
  4. While 循环:

    • WHILE p ≠ ∧ DO
      • 进入一个循环,只要当前节点 p 不为空。
      • PRINT(Data(p)): 打印当前节点的数据。
      • S <= p: 将当前节点 p 压入栈。
      • p ← FirstChild(p): 将 p 移动到其第一个孩子节点。
  5. 后续处理:

    • WHILE p = ∧ AND S 非空 DO
      • 进入一个循环,只有当 p 为空而且栈 S 不为空时。
      • p <= S: 弹出栈顶元素,将其赋给 p
      • p ← NextBrother(p): 将 p 移动到其下一个兄弟节点。
  6. 结束条件:

    • IF S 非空 THEN GOTO NPO3: 如果栈 S 非空,跳转到标签 NPO3,继续遍历。
c. 代码实现
// 先根遍历的非递归算法
void NorecPreOrder(TreeNode* t) {if (t == NULL) {return;}TreeNode* stack[100];  // 假设栈的最大大小为100int top = -1;TreeNode* p = t;while (p != NULL || top != -1) {if (p != NULL) {// 访问当前节点printf("%c ", p->data);// 将当前节点入栈stack[++top] = p;// 移动到当前节点的第一个孩子p = getFirstChild(p);} else {// 出栈并移动到下一个兄弟节点p = getNextBrother(stack[top--]);}}
}
  1. 参数:

    • t: 树的根节点。
  2. 局部变量:

    • stack[100]: 用于模拟栈的数组,存储待访问的节点。
    • top: 栈顶指针,表示栈的当前位置。
  3. 算法过程:

    • 如果树的根节点为空 (t == NULL),直接返回。
    • 初始化当前节点指针 p 为树的根节点 t
    • 使用循环遍历整个树结构,直到当前节点 p 为空且栈 stack 为空。
    • 在循环中:
      • 如果当前节点 p 不为空:
        • 访问当前节点的数据:printf("%c ", p->data);
        • 将当前节点入栈:stack[++top] = p;
        • 移动到当前节点的第一个孩子:p = getFirstChild(p);
      • 如果当前节点 p 为空:
        • 出栈并移动到下一个兄弟节点:p = getNextBrother(stack[top--]);
    • 循环结束后,遍历完成。
  4. 栈的作用:

    • 使用栈来模拟递归调用过程,确保每个节点都能被正确地访问。
    • 入栈操作保存了当前节点的信息,以便在遍历完当前节点的子树后返回到其兄弟节点。
    • 这个算法的时间复杂度是 O(n),其中 n 是树的节点数量。

3. 代码整合

#include <stdio.h>
#include <stdlib.h>// 定义树节点
typedef struct TreeNode {char data;struct TreeNode* firstChild;struct TreeNode* nextBrother;
} TreeNode;// 创建树节点
TreeNode* createNode(char data) {TreeNode* newNode = (TreeNode*)malloc(sizeof(TreeNode));if (newNode != NULL) {newNode->data = data;newNode->firstChild = NULL;newNode->nextBrother = NULL;}return newNode;
}// 释放树节点及其子树
void freeTree(TreeNode* root) {if (root != NULL) {freeTree(root->firstChild);freeTree(root->nextBrother);free(root);}
}// 算法GFC:获取大儿子结点
TreeNode* getFirstChild(TreeNode* p) {if (p != NULL && p->firstChild != NULL) {return p->firstChild;}return NULL;
}// 算法GNB:获取下一个兄弟结点
TreeNode* getNextBrother(TreeNode* p) {if (p != NULL && p->nextBrother != NULL) {return p->nextBrother;}return NULL;
}/* 使用已知的getFirstChild和getNextBrother函数实现先根遍历以t为根指针的树。*/
void PreOrder(TreeNode* t) {// 基本条件检查if (t == NULL) {return;}// 打印当前树节点的数据printf("%c ", t->data);// 递归调用子树的先根遍历TreeNode* child = getFirstChild(t);while (child != NULL) {PreOrder(child);// 迭代调用右兄弟节点的先根遍历child = getNextBrother(child);}
}// 先根遍历的非递归算法
void NorecPreOrder(TreeNode* t) {if (t == NULL) {return;}TreeNode* stack[100];  // 假设栈的最大大小为100int top = -1;TreeNode* p = t;while (p != NULL || top != -1) {if (p != NULL) {// 访问当前节点printf("%c ", p->data);// 将当前节点入栈stack[++top] = p;// 移动到当前节点的第一个孩子p = getFirstChild(p);} else {// 出栈并移动到下一个兄弟节点p = getNextBrother(stack[top--]);}}
}int main() {// 构建左儿子右兄弟链接结构的树TreeNode* A = createNode('A');TreeNode* B = createNode('B');TreeNode* C = createNode('C');TreeNode* D = createNode('D');TreeNode* E = createNode('E');TreeNode* F = createNode('F');A->firstChild = B;B->nextBrother = C;C->nextBrother = D;C->firstChild = E;E->nextBrother = F;// 使用递归先根遍历算法printf("Recursive Preorder: \n");PreOrder(A);printf("\n");// 使用非递归先根遍历算法printf("Non-recursive PreOrder: \n");NorecPreOrder(A);printf("\n");// 释放树节点freeTree(A);return 0;
}

在这里插入图片描述

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

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

相关文章

<b><strong>,<i><em>标签的区别

1. b标签和strong标签 b标签&#xff1a;仅仅是UI层面的加粗样式&#xff0c;并不具备HTML语义 strong标签&#xff1a;不仅是在UI层面的加粗样式&#xff0c;具备HTML语义&#xff0c;表示强调 2. i标签和em标签 i 标签&#xff1a;仅仅是UI层面的斜体样式&#xff0c;并不…

HTML5学习系列之实用性标记

HTML5学习系列之实用性标记 前言实用性标记高亮显示进度刻度时间联系信息显示方向换行断点标注 总结 前言 学习记录 实用性标记 高亮显示 mark元素可以进行高亮显示。 <p><mark>我感冒了</mark></p>进度 progress指示某项任务的完成进度。 <p…

(c语言进阶)内存函数

一.memcpy(void* dest,void* src,int num) &#xff0c;操作单位为字节&#xff0c;完成复制且粘贴字符串 1.应用 #include <stdio.h> #include<string.h> int main() {int arr1[] { 1,2,3,4,5,6,7,8,9,10 };int arr2[20] { 0 };memcpy(arr2, arr1, 20);//从…

Linux - 用户级缓冲区和系统缓冲区 - 初步理解Linux当中文件系统

前言 文件系统 我们先来看两个例子&#xff1a; 这个程序输出&#xff1a; 此时的输出也满足的我们预期。 我们也可以把 程序执行结果&#xff0c;输出重定向到 一个文件当中: 当我们在代码的结尾处&#xff0c;创建了子进程&#xff0c;那么输出应该还是和上述是一样的&…

Day35力扣打卡

打卡记录 相邻字符不同的最长路径&#xff08;树状DP&#xff09; 链接 若节点也存在父节点的情况下&#xff0c;传入父节点参数&#xff0c;若是遍历到父节点&#xff0c;直接循环里 continue。 class Solution:def longestPath(self, parent: List[int], s: str) -> in…

基于Vue+SpringBoot的医院门诊预约挂号系统 开源项目

项目编号&#xff1a; S 033 &#xff0c;文末获取源码。 \color{red}{项目编号&#xff1a;S033&#xff0c;文末获取源码。} 项目编号&#xff1a;S033&#xff0c;文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 功能性需求2.1.1 数据中心模块2.1.2…

数电实验-----实现74LS153芯片扩展为8选1时间选择器以及应用(Quartus II )

目录 一、74LS153芯片介绍 管脚图 功能表 二、4选1选择器扩展为8选1选择器 1.扩展原理 2.电路图连接&#xff08;Quartus II &#xff09; 3.仿真结果 三、8选1选择器的应用 1.三变量表决器 2.奇偶校验电路 一、74LS153芯片介绍 74ls153芯片是属于四选一选择器的芯片。…

【运维篇】Redis 性能测试工具实践

文章目录 1. 前言2. Redis性能测试工具2.1 Redis-benchmark2.1.1 何为Redis-benchmark2.1.2 Redis-benchmark的特点2.1.3 如何使用Redis-benchmark可选的参数 2.2 Memtier_benchmark2.2.1 何为Memtier_benchmark2.2.2 Memtier_benchmark的特点2.2.3 如何使用Memtier_benchmark …

【AI视野·今日Robot 机器人论文速览 第六十五期】Mon, 30 Oct 2023

AI视野今日CS.Robotics 机器人学论文速览 Mon, 30 Oct 2023 Totally 18 papers &#x1f449;上期速览✈更多精彩请移步主页 Daily Robotics Papers Gen2Sim: Scaling up Robot Learning in Simulation with Generative Models Authors Pushkal Katara, Zhou Xian, Katerina F…

网工内推 | 国企、港企网工,年底双薪,NA以上认证即可

01 中航期货有限公司 招聘岗位&#xff1a;信息技术部-网络工程师 职责描述&#xff1a; 1、负责总部、分支机构、外联单位网络的日常运维、故障和应急处置&#xff0c;特别是定期监测设备的运行状态&#xff0c;对存在隐患的地方及时发现改正&#xff0c;保持网络稳定通畅&am…

React 中 react-i18next 切换语言( 项目国际化 )

背景 平时中会遇到需求&#xff0c;就是切换语言&#xff0c;语种等。其实总的来说都是用i18n来实现的 思路 首先在项目中安装i18n插件&#xff0c;然后将插件引入到项目&#xff0c;然后配置语言包&#xff08;语言包需要你自己来进行配置&#xff0c;自己编写语言包&#xff…

接口自动化项目落地之HTTPBin网站

原文&#xff1a;https://www.cnblogs.com/df888/p/16011061.html 接口自动化项目落地系列 找个开源网站或开源项目&#xff0c;用tep实现整套pytest接口自动化项目落地&#xff0c;归档到电子书&#xff0c;作为tep完整教程的项目篇一部分。自从tep完整教程发布以后&#…

C语言:结构体

目录 结构体类型的声明 匿名结构体 全局结构体变量 嵌套结构体 访问结构体成员 结构的自引用 结构体变量的定义和初始化 结构体内存对齐 结构体内存对齐规则 修改默认对齐数 #pragma pack(n) offsetof 求结构体成员相对于结构体开头的偏移量的宏。 为什么存在内存…

pip list 和 conda list的区别

PS : 网上说conda activate了之后就可以随意pip了 可以conda和pip混用 但是安全起见还是尽量用pip 这样就算activate了&#xff0c;进入base虚拟环境了 conda与pip的区别 来源 Conda和pip通常被认为几乎完全相同。虽然这两个工具的某些功能重叠&#xff0c;但它们设计用于不…

硬件开发笔记(十二):RK3568底板电路电源模块和RTC模块原理图分析

若该文为原创文章&#xff0c;转载请注明原文出处 本文章博客地址&#xff1a;https://hpzwl.blog.csdn.net/article/details/134429973 红胖子网络科技博文大全&#xff1a;开发技术集合&#xff08;包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬…

庖丁解牛:NIO核心概念与机制详解 06 _ 连网和异步 I/O

文章目录 Pre概述异步 I/OSelectors打开一个 ServerSocketChannel选择键内部循环监听新连接接受新的连接删除处理过的 SelectionKey传入的 I/O回到主循环 Pre 庖丁解牛&#xff1a;NIO核心概念与机制详解 01 庖丁解牛&#xff1a;NIO核心概念与机制详解 02 _ 缓冲区的细节实现…

如何将本地Portainer管理界面结合cpolar内网穿透工具实现远程浏览器访问

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 Portainer 是一个轻量级的容器管理工具&#xff0c;可以通过 Web 界面对 Docker 容器进行管理和监控。它提供了可…

Android修行手册-POI操作中文API文档

Unity3D特效百例案例项目实战源码Android-Unity实战问题汇总游戏脚本-辅助自动化Android控件全解手册再战Android系列Scratch编程案例软考全系列Unity3D学习专栏蓝桥系列ChatGPT和AIGC &#x1f449;关于作者 专注于Android/Unity和各种游戏开发技巧&#xff0c;以及各种资源分…

redis三种集群方式

redis有三种集群方式&#xff1a;主从复制&#xff0c;哨兵模式和集群。 1.主从复制 主从复制原理&#xff1a; 从服务器连接主服务器&#xff0c;发送SYNC命令&#xff1b; 主服务器接收到SYNC命名后&#xff0c;开始执行BGSAVE命令生成RDB文件并使用缓冲区记录此后执行的所…

键盘方向键移动当前选中的table单元格,并可以输入内容

有类似于这样的表格&#xff0c;用的<table>标签。原本要在单元格的文本框里面输入内容&#xff0c;需要用鼠标一个一个去点以获取焦点&#xff0c;现在需要不用鼠标选中&#xff0c;直接用键盘的上下左右来移动当前正在输入的单元格文本框。 const currentCell React.u…