当前位置: 首页 > news >正文

二叉树的链式结构实现

前言

     该篇是在二叉树介绍及堆-CSDN博客的基础上的。该篇会有点抽象大家要自己多画画图自己感受一下。现在我们开始吧!

     在学习二叉树基本操作时,我们需要先有一个现成的二叉树。来方便我们练习。因为现在我们对二叉树的理解也并不是很深入。在这里创建一个树是方便让我们理解。等我们学的差不多的时候,我们来真正的创建二叉树。

     下图是我创建二叉树的结构。

ed4959117af848a1a6dd0aa55f405c2f.png

typedef int BTDataType;typedef struct BinaryTreeNode
{BTDataType date;struct BinaryTreeNode* leftchild;struct BinaryTreeNode* rightchild;
}BTNode;BTNode* CreateNode(BTDataType x)
{BTNode* node = (BTNode*)malloc(sizeof(BTNode));if (node == NULL){perror("malloc");exit(-1);}node->date = x;node->leftchild = node->rightchild = NULL;return node;
}
//手搓二叉树
BTNode* HandRub()
{BTNode* node1 = CreateNode(1);BTNode* node2 = CreateNode(2);BTNode* node3 = CreateNode(3);BTNode* node4 = CreateNode(4);BTNode* node5 = CreateNode(5);BTNode* node6 = CreateNode(6);node1->leftchild = node2;node2->leftchild = node3;node1->rightchild = node4;node4->leftchild = node5;node4->rightchild = node6;return node1;
}

     在二叉树的介绍及堆这篇文章中的二叉树概念可以发现二叉树定义是递归式的,因此下面操作中基本都是按照该概念实现的。

遍历

     二叉树遍历(Traversal)是按照某种特定的规则,依次对二叉树中的结点进行相应的操作,并且每个结点只操作一次。

二叉树的遍历有三种。

  1. 前序遍历:先访问根节点,在遍历左子树,后遍历右子树。
  2. 中序遍历:先遍历左子树,在访问根节点,后遍历右子树。
  3. 后序遍历:先遍历左子树,在遍历右子树,后访问根节点。

     在遍历中假设遇到空指针则打印N。

前序遍历

     根据前面提到的前序遍历,可以知道会先访问1节点,然后在遍历1节点的左孩子也就是2节点。走到2节点时要重新进行前序遍历,所以会先访问2节点,然后在遍历2节点的左孩子节点也就是3节点。然后在访问3节点,然后遍历3节点的左孩子,但3节点没有左孩子。根据前序遍历它会遍历3节点的右孩子,但3节点没有右孩子。这时它会返回到2节点,遍历2节点的右孩子(2节点本身和左孩子都已经访问过了,根据前序遍历规则它会遍历2节点的右孩子)。2节点没有右孩子,它会返回到1节点的右孩子,然后在次遍历。右边与左边同理。

a1580e5bdcf44ad3871e9f59d2e3f9a9.png

     不要看的它很复杂,实际上思想还是挺简单的。 

     上面遍历完之后的结果应该是:1 2 3 N N N 4 5 N N 6 N N。

     上面说的就很符合递归的特点,把一个大问题拆成与原问题相似,但规模较小的子问题。

//前序遍历
void preOrder(BTNode* ret)
{if (ret == NULL)//判断传入节点是否为空指针{printf("N ");return;}printf("%d ", ret->date);preOrder(ret->leftchild);preOrder(ret->rightchild);
}

     这里的ret == NULL 有两个作用,在下面提供的代码中类似这段的都有该功能

  1. 判断传过来的树是不是空树。
  2. 当左子树或右子树遍历完之后开始回归。

     验证一下我们上面自己写出的结果。

83e611c5643143c4a3ae93cbe2c5149e.png

     可以发现我们自己写的和运行出来的结果没区别。这么复杂的过程用递归实现竟然这么简短,这就是递归的魅力!

中序遍历

     中序遍历和前序遍历原理是一样的,差别只是访问的次序不同,体现在代码上就上顺序的差异。

//中序遍历
void InOrder(BTNode* ret)
{if (ret == NULL){printf("N ");return;}InOrder(ret->leftchild);printf("%d ", ret->date);InOrder(ret->rightchild);
}

后序遍历

     同样的,后序遍历和前序遍历原理是一样的,差别只是访问的次序不同,体现在代码上就上顺序的差异。

//后序遍历
void PostOrder(BTNode* ret)
{if (ret == NULL){printf("N ");return;}PostOrder(ret->leftchild);PostOrder(ret->rightchild);printf("%d ", ret->date);
}

用遍历结果推树的结构

     已知1 2 3 4 5 7 6是前序遍历结果,3 2 1 5 7 4 6是中序遍历结果。求该二叉树的结构。

     由前序遍历结果可知1是根节点,由中序遍历规则可以分开该树的左子树和右子树。

d0cb2f937bb540dca47fcd1b9b8587ca.png

     左子树结构:看前序遍历1之后是2,由前序遍历的特点先根在左后右,得2是1的左孩子,由前序遍历的特点得3是2的左孩子,3由于下一个是4它在右子树中,所以2没有右孩子且3是叶子。

     右子树结构:因为下一个是4,所以4是1的右孩子。由中序遍历的先左在根后右。可以判断4的左子树是57,右子树是6。前序遍历中4之后是5,所以5是4的左孩子

473ffacc765b4152952cda70971b88b5.png

7是5的右孩子。为什么?如果7是5的左孩子那么在前序遍历中7会在5之前。

     最后的结构如下:

580b5b64185444729707bbfa3afdb4b6.png

     如果没看懂上面写的那就对二叉树遍历理解不太好。

节点个数

总节点数

     最简单的方式就是传一个Size来计数。然后用遍历一遍即可。要判断传入节点是否为空指针。

     注意:这里要传Size的地址(函数每次调用会开辟一个栈帧,里面会存放函数的内容,其中就有Size。当函数调用完后栈帧会被销毁掉,Size保留的数据会被销毁,如果这样Size就无法起到它的作用)。

void TreeSize(BTNode* root, int* Size)
{if (root == NULL)return;(*Size)++;TreeSize(root->leftchild, Size);TreeSize(root->rightchild, Size);
}

     上面的写法可以实现目的,那能不能不创建Size来实现呢?其实是可以的。

int TreeSize(BTNode* root)
{return root == NULL ? 0 : TreeSize(root->leftchild) + TreeSize(root->rightchild) + 1;
}

     我在这里就只写了左子树,右子树与它相似大家可以自己试试。大家要结合图好好的理解感受一下。

8c1a98a3259b432fb75dc674bbcbca2b.png

     

叶子节点数

     什么时候是叶子节点呢?

     大家想一下叶子节点有什么特点,它没有孩子这就说明了它的左孩子和右孩子是相同的(在创建节点时默认左孩子和右孩子都为NULL)。这就是判断是否为叶节点的判断条件。

    求叶子节点个数和求节点个数的思想是一致的。其实也就遍历一遍,只不过加了额外的判断条件。

     要判断传入节点是否为空指针。如果判断条件如果成立就返回1,否则就继续。

int TreeLeafSize(BTNode* root)
{if (root == NULL)return 0;return root->leftchild == root->rightchild ? 1 : TreeLeafSize(root->leftchild) + TreeLeafSize(root->rightchild);
}

第k层节点数

     在这里要先认定第一层的高度为1不是0(有些书上会把第一层的高度认为是0)。

     假设我们要第二层的节点数。

c317589c38e94cbe9a41e229671cf67c.png

     由上图可知,当k == 1时即到了该层 。其它的就和上面的求叶子节点个数一样了。

     我们需要的是第1层,当k小于1就没必要在继续了返回就行了。还有要判断传入节点是否为空指针。

int TreeNodeSize(BTNode* root, int k)
{if (k < 1 || root == NULL)return 0;return k == 1 ? 1 : TreeNodeSize(root->leftchild, k - 1) + TreeNodeSize(root->rightchild, k - 1);
}

查找值为x的节点

     先找根节点再在左子树中找后在右子树中找。这和前序遍历很像。只是在前序遍历的基础上加了点限制条件。

     在遍历左子树和右子树时,要记录一下返回值,别你找到了没带回来,这就尴尬了。

     要判断传入的节点是否为空。遇到要找的值那就一直返回,直到跳出该函数。在此之前要加个判断返回的该节点不为NULL。如果没找到那就返回NULL。

BTNode* TreeFind(BTNode* root, int x)
{if (root == NULL)return NULL;if (root->date == x)return root;BTNode* ret1 = TreeFind(root->leftchild, x);if (ret1)return ret1;BTNode* ret2 = TreeFind(root->rightchild, x);if (ret2)return ret2;return NULL;
}

创建二叉树

     好了,经过上面知识的学习相信大家对二叉树有了一定的理解,现在让我们创建二叉树吧!

     我们要创建一个数组,用来存放值和空指针。为什么要存放空指针呢?如果不存放那就有可能是任意一种遍历。在这里我将用#来表示空指针。传入的是数组,数组需要下标来找数据,所以在传参时要传下标的地址(其原因和求节点数传地址的道理是一样的)。

     这里我用输入是前序遍历的情况来表示(下面的代码仅适用于前序遍历输入)。

     要先判断是否为空,后让下标加加,加加时不要写在判断条件上,因为如果不是空的话在判断之后下标会加加那就少了一个值。然后创建节点再把值放进去后加加,然后构建左子树怎么做呢?让节点的左孩子调用该函数即可。右子树的构建同理。最后返回创建节点即可。

BTNode* CreateTree(char* arr, int* i)
{if(arr[*i] == '#'){(*i)++;return NULL;}TNode* root = (TNode*)malloc(sizeof(TNode));root->val = arr[(*i)++];root->left = CreateTree(arr, i);root->right = CreateTree(arr, i);return root;
}

相关文章:

二叉树的链式结构实现

前言 该篇是在二叉树介绍及堆-CSDN博客的基础上的。该篇会有点抽象大家要自己多画画图自己感受一下。现在我们开始吧&#xff01; 在学习二叉树基本操作时&#xff0c;我们需要先有一个现成的二叉树。来方便我们练习。因为现在我们对二叉树的理解也并不是很深入。在这里创建一个…...

MySQL远程连接

文章目录 MySQL远程连接(Linux)一、更改MySQL配置文件二、进入MySQL修改用户表host值三、使用其他电脑即可远程访问数据库MySQL远程连接(Linux)一、修改my.ini中的配置文件二、修改用户权限三、远程连接 MySQL远程连接(Linux) 以下MySQL远程连接&#xff1a;MySQL部署环境为Ubu…...

奔驰大G升级电动踏板效果

奔驰大G车型的升级旋转电动踏板是一项非常实用的功能&#xff0c;它为驾驶者提供了诸多便利和舒适性。以下是关于这一功能的实用性介绍&#xff1a; 便利的上下车体验&#xff1a;旋转电动踏板可以在车辆停稳的情况下自动伸出&#xff0c;为乘客提供便利的上下车体验。特别是对…...

【xilinx】vivado中的xpm_cdc_gray.tcl的用途

背景 【Xilinx】vivado methodology检查中出现的critical Warning-CSDN博客 接上篇文章&#xff0c;在vivado进行 methodology检查时出现了严重警告&#xff0c;顺着指示查到如下一些问题 TIMING #1 Warning An asynchronous set_clock_groups or a set_false path (see con…...

windows中安装zookeeper

https://zhuanlan.zhihu.com/p/692451839 【zookeeper】在Windows上启动zookeeper_windows启动zk-CSDN博客 Index of /apache/zookeeper/zookeeper-3.9.2 Index of /apache/zookeeper/zookeeper-3.9.2 Zookeeper的应用场景 1、配置管理 2、服务注册中心 3、主从协调 4、…...

直接写和放在函数中不同的R语言用法

索引数据框中的某一列 df$A可以索引数据框df中列名为A的列的所有值。那么假如列名是一个R对象怎么做&#xff1f; df <- data.frame(A1:5, B(1:5)*2)df$A## [1] 1 2 3 4 5needed_column A# df$needed_column ? Wrong# 注意是双方括号 df[[needed_column]]## [1] 1 2 3 4…...

《mysql轻松学习·二》

1、创建数据表 contacts&#xff1a;数据表名 auto_increament&#xff1a;自动增长 primary key&#xff1a;主键 engineInnoDB default charsetutf8; 默认字符集utf8&#xff0c;不写就默认utf8 对数据表的操作&#xff1a; alter table 数据表名 add sex varchar(1); //添…...

Swift对比版本号

在 Swift 中比较两个版本号的大小可以使用以下方法: func compareVersions(_ version1: String, _ version2: String) -> ComparisonResult {let v1Components version1.components(separatedBy: ".")let v2Components version2.components(separatedBy: "…...

MySQL数据表的“增删查改“

我们学习数据库, 最重要的就是要学会对数据表表进行"增删查改"(CRUD).(C -- create, R -- retrieve, U -- update, D -- delete) 目录 一. "增"(create) 1. 普通新增 2. 指定列新增 3. 一次插入多行 4. 用insert插入时间 5. 小结 二. "查"…...

Github查询语法

转载自link 基础查询结构 一个关键词会匹配文件内容或文件路径。 多个关键词会匹配文件内容&#xff0c;只要包含关键词&#xff0c;就会出现在搜索结果中&#xff0c;不论前后顺序&#xff0c;是否是一个单词&#xff08;多个关键词之间没有空格&#xff09;。 还可以使用…...

pqgrid的使用

npm安装pqgrid npm install pqgridf --registryhttps://registry.npmmirror.com npm install jquery-ui --registryhttps://registry.npmmirror.comvue文件 <template><div><div id"grid_json"></div></div> </template><s…...

媳妇面试了一家公司,期望月薪20K,对方没多问就答应了,只要求3天内到岗,可我总觉得哪里不对劲。

“20k&#xff01;明天就来上班吧&#xff01;” 听到这句话&#xff0c;你会不会两眼放光&#xff0c;激动得差点跳起来&#xff1f; 朋友媳妇小丽&#xff0c;最近就经历了这样一场“梦幻面试”。然而&#xff0c;事情的发展却远没有想象中那么美好…… “这公司也太好了吧…...

【Makefile笔记】小白入门篇

【Makefile笔记】小白入门篇 文章目录 【Makefile笔记】小白入门篇所需组件一、简单了解Makefile1.Makefile简介2.Makefile 原理 二、为什么要使用Makefile1.解决编译时链库的不便2.提高编译效率&#xff0c;缩短编译时间&#xff08;尤其是大工程&#xff09; 三、Makefile语法…...

快速入门文件操作+5种例子演示

文件操作 基本操作注意事项例子1&#xff1a;读取文件内容例子2&#xff1a;写入文件内容例子3&#xff1a;追加文件内容例子4&#xff1a;读取并写入文件内容&#xff08;复制文件&#xff09;例子5&#xff1a;使用二进制模式读写文件 基本操作 在C语言中&#xff0c;使用文…...

基于Vue3的Uniapp实训项目|一家鲜花店

基于Vue的Uniapp实训指导项目 项目预览&#xff1a; 在这里插入图片描述 pages.json {"pages": [ //pages数组中第一项表示应用启动页&#xff0c;参考&#xff1a;https://uniapp.dcloud.io/collocation/pages{"path": "pages/index/index",&…...

Python3 字典

前言 本文主要介绍Python中的字典&#xff08;dict&#xff09;,主要内容包括&#xff1a;字典简介、字典特性、字典的基本操作。 文章目录 前言一、字典简介二、字典特性1、键值对2、无序性?3、可变性4、键的唯一性5、值的类型不限 三、字典的基本操作1、创建2、访问3、增加…...

JPA详解

文章目录 JPA概述JPA的优势JPA注解 JPA概述 Java Persistence API&#xff08;JPA&#xff09;是 Java EE 平台的一部分&#xff0c;它为开发者提供了一种用于对象关系映射&#xff08;ORM&#xff09;的标准化方法。JPA 提供了一组 API 和规范&#xff0c;用于在 Java 应用程…...

Linux线程:线程分离

目录 一、什么是线程分离 1.1pthread_detach 1.2pthread线程库存在的意义 1.3__thread线程的局部存储 1.4系统调用clone 一、什么是线程分离 1.1pthread_detach 默认情况下&#xff0c;新创建的线程是joinable的&#xff0c;线程退出后&#xff0c;需要对其进行pthread_joi…...

chatgpt之api的调用问题

1.调用api过程中&#xff0c;出现如下报错内容 先写一个测试样例 import openaiopenai.api_key "OPEN_AI_KEY" openai.api_base"OPEN_AI_BASE_URL" # 是否需要base根据自己所在地区和key情况进行completion openai.ChatCompletion.create(model"g…...

Java中lambda表达式是啥怎么使用

在Java中&#xff0c;Lambda表达式&#xff08;也称为闭包&#xff09;是一种简洁地表示匿名函数&#xff08;即没有名称的函数&#xff09;的方式。它们允许你将函数作为参数传递或赋值给变量&#xff0c;从而简化代码。Lambda表达式在Java 8及更高版本中引入。 Lambda表达式…...

服务器硬防的应用场景都有哪些?

服务器硬防是指一种通过硬件设备层面的安全措施来防御服务器系统受到网络攻击的方式&#xff0c;避免服务器受到各种恶意攻击和网络威胁&#xff0c;那么&#xff0c;服务器硬防通常都会应用在哪些场景当中呢&#xff1f; 硬防服务器中一般会配备入侵检测系统和预防系统&#x…...

基于当前项目通过npm包形式暴露公共组件

1.package.sjon文件配置 其中xh-flowable就是暴露出去的npm包名 2.创建tpyes文件夹&#xff0c;并新增内容 3.创建package文件夹...

关键领域软件测试的突围之路:如何破解安全与效率的平衡难题

在数字化浪潮席卷全球的今天&#xff0c;软件系统已成为国家关键领域的核心战斗力。不同于普通商业软件&#xff0c;这些承载着国家安全使命的软件系统面临着前所未有的质量挑战——如何在确保绝对安全的前提下&#xff0c;实现高效测试与快速迭代&#xff1f;这一命题正考验着…...

让回归模型不再被异常值“带跑偏“,MSE和Cauchy损失函数在噪声数据环境下的实战对比

在机器学习的回归分析中&#xff0c;损失函数的选择对模型性能具有决定性影响。均方误差&#xff08;MSE&#xff09;作为经典的损失函数&#xff0c;在处理干净数据时表现优异&#xff0c;但在面对包含异常值的噪声数据时&#xff0c;其对大误差的二次惩罚机制往往导致模型参数…...

在 Spring Boot 项目里,MYSQL中json类型字段使用

前言&#xff1a; 因为程序特殊需求导致&#xff0c;需要mysql数据库存储json类型数据&#xff0c;因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...

STM32标准库-ADC数模转换器

文章目录 一、ADC1.1简介1. 2逐次逼近型ADC1.3ADC框图1.4ADC基本结构1.4.1 信号 “上车点”&#xff1a;输入模块&#xff08;GPIO、温度、V_REFINT&#xff09;1.4.2 信号 “调度站”&#xff1a;多路开关1.4.3 信号 “加工厂”&#xff1a;ADC 转换器&#xff08;规则组 注入…...

Linux-进程间的通信

1、IPC&#xff1a; Inter Process Communication&#xff08;进程间通信&#xff09;&#xff1a; 由于每个进程在操作系统中有独立的地址空间&#xff0c;它们不能像线程那样直接访问彼此的内存&#xff0c;所以必须通过某种方式进行通信。 常见的 IPC 方式包括&#…...

2025.6.9总结(利与弊)

凡事都有两面性。在大厂上班也不例外。今天找开发定位问题&#xff0c;从一个接口人不断溯源到另一个 接口人。有时候&#xff0c;不知道是谁的责任填。将工作内容分的很细&#xff0c;每个人负责其中的一小块。我清楚的意识到&#xff0c;自己就是个可以随时替换的螺丝钉&…...

Springboot 高校报修与互助平台小程序

一、前言 随着我国经济迅速发展&#xff0c;人们对手机的需求越来越大&#xff0c;各种手机软件也都在被广泛应用&#xff0c;但是对于手机进行数据信息管理&#xff0c;对于手机的各种软件也是备受用户的喜爱&#xff0c;高校报修与互助平台小程序被用户普遍使用&#xff0c;为…...

基于Python的气象数据分析及可视化研究

目录 一.&#x1f981;前言二.&#x1f981;开源代码与组件使用情况说明三.&#x1f981;核心功能1. ✅算法设计2. ✅PyEcharts库3. ✅Flask框架4. ✅爬虫5. ✅部署项目 四.&#x1f981;演示效果1. 管理员模块1.1 用户管理 2. 用户模块2.1 登录系统2.2 查看实时数据2.3 查看天…...