数据结构05:树的定义与双亲表示法[持续更新中]

news2025/1/13 6:19:40

参考用书:王道考研《2024年 数据结构考研复习指导》

参考用书配套视频:5.1.1 树的定义和基本术语_哔哩哔哩_bilibili

特别感谢: Chat GPT老师[部分名词解释、修改BUG]、BING老师[封面图]~

备注:博文目前是未完成的状态,如题,目前只写了树的存储结构~~本博文篇幅较长,笔者无法一次全部更新,毕竟我的编程水平就是个两脚Bug生成兽,经常1段代码可以卡1天,完成时间初步预计为23.06.17~ 🥲

考研笔记整理,内容预计包含树、二叉树与森林的基本概念、存储结构,构造与遍历、树、森林与二叉树的转换,代码为C++~考研一起加油~ 🫡

第1版:查资料、写BUG、画导图、画配图ing~


目录

目录

目录

思维导图

树的概念

树的基本术语

树的基本性质

树的存储结构

双亲表示法

结语

备注:此部分待二叉树、森林部分的博文完成后补充~


思维导图


树的概念

树的定义:树是n(n≥0)个结点的有限集。当n=0时,称为空树。在任意一棵非空树应满足:

  • 有且仅有一个特定的称为根的结点。
  • 当n>1时,其余结点可分为m(m>0)个互不相交的有限集T1,T2,...,Tm,其中每个集合本身又是一棵树,并且称为根的子树。 //因此,树是递归的数据结构

树的基本术语

图:树的图形表示+术语注释 

 (1)结点、度

  • 结点:一种数据结构,包含数据和对一个或多个其他节点的引用(例如:指针)。
    • 根结点:树的根节点没有前驱,除根结点以外的所有结点有且仅有1个前驱;
    • 分支结点:除根节点外,度>0的结点称为分支结点;
    • 分支结点:除根节点外,度 = 0的结点称为分支结点;
  • 结点关系:
    • 祖先结点:从根结点到指定结点的路径上的所有结点,例如结点A、B、E均为结点K的祖先结点;
    • 子孙结点:从指定结点到叶子点的路径上的所有结点,例如结点K、L、F均为结点B的子孙结点;
    • 双亲结点:祖先结点中最接近指定结点的结点,例如结点E是结点K、L的双亲结点。
    • 孩子结点:子孙结点中最接近指定结点的结点,例如结点E、F是结点B的孩子结点。
    • 兄弟结点:具有相同父节点的两个结点,例如结点K与结点L是兄弟结点。
  • 度数:
    • 结点的度:树中一个结点的孩子个数称为度数。
    • 树的度:树中结点的最大度数称为数的度。例如图中结点的度数为3,因此这棵树的结点就是3。

(2)层次、深度、高度:

  • 层次:一个节点的层级是从根节点到该节点的边数。
  • 高度:树中结点的最大层数。

(3)路径:

  • 路径:两个结点之间的路径是由这两个结点之间所经过的结点序列构成的;
  • 结点的路径长度:两个指定结点路径上经过的边的个数。
  • 树的路径长度:树根到每个结点的路径长度的总和。

(4)有序、无序:

  • 有序树:是节点按特定顺序排列的树,结点之间不能互换,例如二叉搜索树。
  • 无序树:是指节点未按任何特定顺序排列的树。

树的基本性质

(1)结点与结点、结点与边

  • n个结点的树 有 n-1条边;// 根节点无前驱,因此无指向根结点的边~
  • 树中的结点数 - 1 =\sum所有结点的度数; // 结点的度数代表孩子结点的个数,根节点为特殊的无前驱的结点,因此需要 -1;

(2)结点与度

  • 度为m的树,在第 i 层上结点数 ≤ m^(i-1); // 按照结点的度均为m的情况考虑,第1层最多有m^(1-1)=1个结点,第2层最多有m^(2-1)=m个结点...递推可求~

(3)结点与高

  • 高度为h的m叉树 结点数 n ≤ (m^h -1)/(m-1); // 等比公式可求,Sn=首项(1-公比的n次方)/(1-公比)

嗯,这个我们以满3叉树(m=3)举栗:

结点与高度的公式推算
层数(h)本层最多结点数首层累加至本层结点数
3叉树第1层m^(h-1)=3^(1-1)= 13^0 = 1
3叉树第2层m^(h-1)=3^(2-1)= 33^0 + 3^1 = 4
3叉树第3层m^(h-1)=3^(3-1)= 93^0 + 3^1 + 3^2 = 13
3叉树第4层m^(h-1)=3^(4-1)= 273^0 + 3^1 + 3^2+ 3^3 = 40
.........
3叉树第h层m^(h-1)=3^(h-1)

3^0 + 3^1 + 3^2+ ... +3^(h-1)

=  1 x(1- 3^h)/(1-3)

=(3^h -1)/ (3-1)

m叉树第h层m^(h-1)(m^h -1)/ (m-1)

树的存储结构

双亲表示法

描述:这种存储结构采用一组连续空间来存储每个结点,同时在每个结点中增加一个伪指针,指示其双亲结点在数组中的位置。

特点:

  • 简洁直观:相比其它存储方式易于理解与实现。
  • 存储结构:顺序存储和链式存储均可实现,其中顺序存储较为常见。
  • 存取效率:可以很快得到每个结点的双亲结点,但求孩子结点时需要遍历整个结构;不过这并不是硬伤,可以根据需要在结构体中增加一个用于存放孩子结点的伪指针。// 所以这里叫做顺序存储法是不是比双亲存储法更合适一些~~
  • 是否有序:双亲表示法不适合表示有序树,更适合表示无序树,因为无序树中节点的子节点没有明确的顺序关系。

图示:源于《王道》教材图5.14 树的双亲表示法

双亲表示法 顺序存储结构 核心代码:

#define MAX_TREE_SIZE 100   //树中可以存储的结点数

typedef struct{     //树中结点的结构,该结构有两个字段
    ElemType data;   //该字段存储结点的数据元素
    int parent;      //该字段存储结点的双亲指针(伪指针)
}PTNode;

typedef struct{     //树的结构,该结构有两个字段
    PTNode nodes[MAX_TREE_SIZE];   //结构数组,用于存储树中的结点
    int n;                         //树中的结点数
}PTree;

双亲表示法 顺序存储结构 案例:

要求:存储上图中的树,并顺序输出结点~

思路:

LocateElem封装按值查找函数,并寻找双亲与孩子结点~

  1. 用2个整型参数i、j,记录并遍历本结点与节点的位序;
  2. 找到目标结点后,输出当前结点的信息以及结点的信息;

  3. 通过循环寻找并输出孩子结点的信息,同时记录子结点的数量;如果子结点的数量为0,则输出没有子结点的提示信息;[这一步时间开销会很高,仅寻找相邻结点];

  4. 如果遍历到末尾没有找到结点,则输出没有该结点的提示信息。

get封装按为查找函数,并寻找祖先与子孙结点~

  1. 判断树中是为空,如果是,返回错误;如果否,继续执行;
  2. 判断值是否越界,如果是,返回错误;如果否,继续执行;
  3. 输出目标结点的data与parent;
  4. 通过递归寻找并输出祖先/子孙结点的信息,同时记录子结点的数量;如果子结点的数量为0,则输出没有子结点的提示信息;如果父节点已为根结点,则反馈到第2步;[这一步时间开销会很高,可寻找所有祖先/子孙结点]; 

#include <iostream>
#define MAX_TREE_SIZE 100

typedef struct{
    char data;
    int parent;
}PTNode;

typedef struct{
    PTNode nodes[MAX_TREE_SIZE];
    int n;
}PTree;

void InitTree(PTree* tree) {
    for(int i = 0; i < MAX_TREE_SIZE; i++){
        tree->nodes[i].data = 0;
        tree->nodes[i].parent = -1;
    }
    tree->n = 0;
}

void addNode(PTree* tree, char data, int parentIndex) {
    if (tree->n >= MAX_TREE_SIZE) {
        std::cout <<"错误:树已满,不能再添加数据。\n";
        return;
    }

    PTNode newNode;
    newNode.data = data;

    if (parentIndex < 0){
        newNode.parent = -1;
    }else{
        newNode.parent = parentIndex;
    }

    tree->nodes[tree->n] = newNode;
    tree->n++;

}

void LocateElem(const PTree* tree, char data) {
    int i, j;
    int childCount = 0;
    int firstChildPos = -1;

    for (i = 0, j = -1; i < tree->n; i++) {
        if (tree->nodes[i].data == data) {
            std::cout << "找到结点 " << data << std::endl;
            std::cout << "当前结点信息:" << "位置: " << i << ", 数据: " << tree->nodes[i].data << ", 双亲位置: " << tree->nodes[i].parent << std::endl;

            j = tree->nodes[i].parent;
            if (j != -1) {
                std::cout << "父节点信息:" << "位置: " << j << ", 数据: " << tree->nodes[j].data << ", 双亲位置: " << tree->nodes[j].parent << std::endl;
            }

            for (int k = 0; k < tree->n; k++) {
                if (tree->nodes[k].parent == i) {
                    if (childCount == 0) {
                        firstChildPos = k;
                    }
                    std::cout << "子节点信息:" << "位置: " << k << ", 数据: " << tree->nodes[k].data << ", 双亲位置: " << tree->nodes[k].parent << std::endl;
                    childCount++;
                }
            }

            if (childCount == 0) {
                std::cout << "该结点没有子节点" << std::endl;
            } else {
                std::cout << "子节点数量: " << childCount << std::endl;
              //std::cout << "第一个子节点信息:" << "位置: " << firstChildPos << ", 数据: " << tree->nodes[firstChildPos].data << ", 双亲位置: " << tree->nodes[firstChildPos].parent << std::endl;
            }

            return;
        }
    }

    std::cout << "未找到结点 " << data << std::endl;
}


void GetOffspring(const PTree& tree, int index) {
    if (index < 0 || index >= tree.n) {
        std::cout << "错误:索引越界。\n";
        return;
    }

    const PTNode& currentNode = tree.nodes[index];

    std::cout << "结点位置: " << index << ", 数据: " << currentNode.data << ", 父节点位置: " << currentNode.parent << std::endl;

    int childCount = 0;

    for (int i = 0; i < tree.n; i++) {
        if (tree.nodes[i].parent == index) {
            childCount++;
            GetOffspring(tree, i);  // 递归调用时传递孩子结点的索引
        }
    }

    if (childCount == 0) {
        std::cout << "该结点没有孩子结点。\n";
    }

}

void GetAncestors(const PTree& tree, int index) {
    if (index < 0 || index >= tree.n) {
        std::cout << "错误:索引越界。\n";
        return;
    }

    const PTNode& currentNode = tree.nodes[index];

    std::cout << "结点位置: " << index << ", 数据: " << currentNode.data << ", 父节点位置: " << currentNode.parent << std::endl;

    int parentIndex = currentNode.parent;
    if (parentIndex >= 0) {
        GetAncestors(tree, parentIndex);
    }

}

int main() {
    PTree newtree;
    // 初始化树对象
    InitTree(&newtree);

    // 增加节点
    char data;
    int parentIndex;
    while (std::cout << "输入结点: " && std::cin >> data && data != '\\') {
        std::cout << "输入结点的双亲位置: ";
        std::cin >> parentIndex;
        addNode(&newtree, data, parentIndex);
    }

    std::cout << std::endl;

    // 输出结点
    for (int i = 0; i < newtree.n; i++) {
        std::cout << "结点位置: " << i << ", 数据: " << newtree.nodes[i].data << ", 父节点位置: " << newtree.nodes[i].parent << std::endl;
    }

    std::cout << std::endl;

    // 输出按值查找结点信息
    char target1;
    std::cout << "请输入要按值查找的结点: ";
    std::cin >> target1;
    LocateElem(&newtree, target1);

    std::cout << std::endl;

    // 输出子孙结点
    int targetIndex1;
    std::cout << "寻找该位序的子孙结点: ";
    std::cin >> targetIndex1;
    GetOffspring(newtree, targetIndex1);

    std::cout << std::endl;

    // 输出祖先结点
    int targetIndex2;
    std::cout << "寻找该位序的祖先结点: ";
    std::cin >> targetIndex2;
    GetAncestors(newtree, targetIndex2);

    return 0;
}

运行的效果如下图所示:


结语

博文未完待续,写得模糊或者有误之处,欢迎小伙伴留言讨论与批评~😶‍🌫️

码字不易,若有所帮助,可以点赞支持一下博主嘛?感谢~🫡

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

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

相关文章

【数据结构】哈希表(Map和Set)

文章目录 Map和Set模型MapTreeMap和HashMap对比Entry<K,V>常用方法 SetTreeSet和HashSet对比常用方法 OJ练习只出现一次数字复制带随机指针的链表宝石与石头坏键盘打字前K个高频单词 哈希表哈希表所用数据结构解决哈希冲突闭散列开散列 避免哈希冲突哈希函数设计负载因子…

springboot bean的生命周期

Spring Boot是一个非常流行的Java框架&#xff0c;它提供了许多功能&#xff0c;使开发人员可以快速构建和部署应用程序。其中一个非常重要的功能是Spring Boot Bean的生命周期。在本文中&#xff0c;我们将深入探讨Spring Boot Bean的生命周期&#xff0c;以及如何最大化利用它…

光伏发电系统最大功率跟踪控制MATLAB仿真模型(电导增量法+扰动观察法)

光伏发电系统最大功率跟踪控制MATLAB仿真模型&#xff08;电导增量法扰动观察法参考文献&#xff09;资源-CSDN文库https://download.csdn.net/download/weixin_56691527/87878528 模型介绍&#xff1a; 模型主要包含光伏电池模块、直流升压模块、以及最大功率跟踪控制模块。…

深度学习--神经网络全面知识点总结(持续更新中)

文章目录 神经网络基础1.1 什么是神经网络&#xff1f;1.2 神经元和激活函数1.3 前向传播和反向传播1.4 损失函数和优化算法 深度神经网络2.1 卷积神经网络&#xff08;CNN&#xff09;2.2 循环神经网络&#xff08;RNN&#xff09;2.3 长短期记忆网络&#xff08;LSTM&#xf…

C#可视化 商品信息管理系统(具体做法及全部代码)

目录 题目&#xff1a; 效果图&#xff1a; 数据库&#xff1a; 做法&#xff1a; 主页面添加menustrip&#xff0c;之后添加两个窗体双击事件&#xff0c;双击下拉区域就好了。 添加界面 查询按钮功能 datagirdview设置 全部代码&#xff1a; DBHelper类 From1主窗体代…

【并发篇】01 java中的线程状态

Java线程分为6种状态&#xff1a; &#xff08;1&#xff09;新建NEW&#xff1a;用new关键字创建的线程就是新建状态&#xff0c;这时候还没有和系统底层真正的线程关联起来&#xff0c;还仅仅只是一个java对象&#xff0c;所以这个时候这个线程不会被系统分配给cpu。 &#…

【云原生】docker

容器化越来越受欢迎&#xff0c;因为容器是&#xff1a; ●灵活&#xff1a;即使是最复杂的应用也可以集装箱化。 ●轻量级&#xff1a;容器利用并共享主机内核。 ●可互换&#xff1a;可以即时部署更新和升级。 ●便携式&#xff1a;可以在本地构建&#xff0c;部署到云&#…

shel脚本基础1——变量基础、脚本运算符

文章目录 一、变量基础二、bash变量类型三、脚本基础知识3.1 脚本测试命令3.2 shell运算符3.3 命令逻辑关系3.4 bash条件判断3.4.1 条件测试的表达式3.4.2 整数测试3.4.2 字符测试3.4.3 文件测试3.4.4 组合测试条件 一、变量基础 变量的含义&#xff1a; 变量可以通过变量名访问…

5.数据结构期末复习之图以及相关算法

1.应用: 教学计划(先修什么才能修什么课程)社交网络地图导航 2.什么是图? 顶点(有穷非空个)顶点的边 ,抽象为G(V,E) (Vert,Edge) 1.无向图: 点到点都可以到达 表示为(vi,vj) 2.有向图 只能一个点到另外一个点 表示为<vi,vj> 又分为: 1.非带权图(上面的例子就是) 2.带权图…

Day29

TCP_SER #include <myhead.h>//定义处理错误的宏函数 #define ERR_MSG(msg) do{\fprintf(stderr,"line:%d %s %s",__LINE__, __FILE__, __func__);\perror(msg);\ }while(0)//定义IP和端口号 #define IP "192.168.2.186" #define PORT 6666int …

数组的定义方式及访问

问题 如何创建及访问数组。 方法 了解数组的概念 数组就是存储多个数据的容器&#xff0c;数组的长度固定&#xff0c;多个数据的数据类型要一致。 数组的三种定义方式 数据存储的数据类型[] 数组名字 new 数组存储的数据类型[长度] 数据类型[] 数组名 new 数据类型[]{元素…

测试人35岁何去何从?软件测试路在何方?“我“一路升级打怪...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 测试员干到35岁&a…

暑期健身房招生活动文案分享,在线传单设计

想要有效地宣传健身房的信息&#xff0c;还能够吸引学员了解报名的宣传单要怎么制作&#xff1f;不用自己动手设计&#xff0c;借助在线模板和在线设计工具&#xff0c;一键就能生成专属的招生传单。以下是在线招生传单设计教程和传单设计模板使用教程。 首先找到一个在线设计工…

Juypter更改默认路径,并且解决改完默认路径不自动跳转的问题

目录 更改默认路径 第一步先在你所安装的juypter的环境下输入下面代码 第二步在自己的文件夹下找到这个目录&#xff0c;用记事本打开 第三步更改快捷方式 解决浏览器不调转方法 相对于juypter来讲&#xff0c;还是挺好用的&#xff0c;自我感觉比pycharm更好一点。 juypt…

ES6中 Promise和使用场景

介绍 &#x1f959;&#x1f959;&#x1f959;更加合理和更加强大 Promise&#xff0c;译为承诺&#xff0c;是异步编程的一种解决方案&#xff0c;比传统的解决方案&#xff08;回调函数&#xff09;更加合理和更加强大 在以往我们如果处理多层异步操作&#xff0c;我们往往…

群晖服务器被encrypted勒索病毒攻击后的表现与如何解密勒索病毒

群晖服务器是一种高效的数据管理方案&#xff0c;但是如果被Encrypted勒索病毒感染&#xff0c;可能会导致许多重要的数据会丢失。Encrypted勒索病毒将加密被感染服务器上的文件&#xff0c;导致它们无法打开。一旦我们遭到encrypted勒索病毒攻击&#xff0c;建议大家选择专业的…

【Web开发技术】异常处理机制的使用

文章目录 一、引言1、应用场景2、语法3、体系 二、详细设计&#xff08;后端开发&#xff09;1、信息提示常量类2、自定义异常3、Model层4、Controller层5、View层 一、引言 编程也有很久的时间了&#xff0c;如果每个报的错误和寻找的答案写成一张纸&#xff0c;叠加起来也得有…

JAVA项目代码几乎没有改动,重新发布后突然报错,无法启动

项目场景&#xff1a; 提示&#xff1a;这里简述项目相关背景&#xff1a; JAVA项目代码几乎没有改动&#xff0c;重新发布后突然报错&#xff0c;无法启动 问题描述 提示&#xff1a;这里描述项目中遇到的问题&#xff1a; JAVA项目代码几乎没有改动&#xff0c;重新发布后…

2023年PMP 续证的费用和流程最全介绍

PMP证书并不是终身有效的&#xff0c;是需要三年一续的&#xff0c;PMI会在有效期截止前给持证者发邮件提示换证&#xff08;续证&#xff09;。目前的续证费用为150美金和积满60个PDU&#xff0c;60个PDU是在证书三年有效期内积满就可以&#xff0c;像我的培训机构艾威每周都会…

当Mysql缓慢时,这几招可解燃眉之急

第一步定位问题源&#xff1a; 常见的以查询Mysql性能问题的方法 1.大部分的性能问题都是查询过慢的问题&#xff0c;可以查询慢sql日志。 通过慢查询日志定位那些执行效率较低的SQL语句&#xff0c;用–log-slow-queries[ file_name]选项启动时&#xff0c;mysqld写一个包含所…