数据结构初阶 · 链式二叉树的部分问题

news2024/11/26 13:41:26

目录

前言:

1 链式二叉树的创建

2 前序 中序 后序遍历

3 树的节点个数

4 树的高度

5 树的叶子节点个数

6 树的第K层节点个数


前言:

链式二叉树我们在C语言阶段已经实现了,这里介绍的是涉及到的部分问题,比如求树的高度,求树的节点个数等,连接部分就手动连接,用一个样例来介绍涉及到的几个问题。

这里对前面知识反馈比较大的是递归,可以说每个问题都用到了递归。


1 链式二叉树的创建

因为是链式二叉树,所以有两个指针,分别指向右孩子节点和左孩子节点,给上值,手动连接即可:

typedef struct TreeNode
{
	struct TreeNode* left = NULL;
	struct TreeNode* right = NULL;
	int data;
}TreeNode;

TreeNode* Tree()
{

	TreeNode* node1 = new TreeNode;
	TreeNode* node2 = new TreeNode;
	TreeNode* node3 = new TreeNode;
	TreeNode* node4 = new TreeNode;
	TreeNode* node5 = new TreeNode;
	TreeNode* node6 = new TreeNode;
	//TreeNode* node7 = new TreeNode;

	node1->data = 1;
	node2->data = 2;
	node3->data = 3;
	node4->data = 4;
	node5->data = 5;
	node6->data = 6;
	//node7->data = 7;

	node1->left = node2;
	node1->right = node4;
	node2->left = node3;
	node4->left = node5;
	node4->right = node6;
	//node5->left = node7;
	return node1;
}

整体构建出来就是这样,对于初学链式二叉树的同学来说,NULL给上是很有必要的,这对于后面的遍历问题有帮助。


2 前序 中序 后序遍历

前序遍历的顺序是 根 左子树 右子树,中序遍历的的顺序是左子树 根 右子树 ,后序遍历的顺序是左子树,右子树,根。

咱们从前序开始,顺便进行打印,从1开始,到左子树是2,2的左子树是3,3是左子树是NULL,再往下就没了,递归回来是3的右子树,那么此时2的左子树就遍历完成了,2的右子树是NULL,这时候1的左子树就遍历完了,然后是右子树,到4,然后是4的左子树5,接着是5的左右子树,这时4的左子树就遍历完了,然后是右子树,当6遍历完了之后,才是整个树都遍历完了。

那么代码会不会很复杂呢?

不会,递归都有一个特点,思路难想,代码简单:

void PrevOrder(TreeNode* root)
{
	if (root == NULL)
	{
		cout << "N ";
		return;
	}
	cout << root->data << " ";
	PrevOrder(root->left);
	PrevOrder(root->right);
}

这就遍历完了。

打印的结果是1 2 3 N N N 4 5 N N 6 N N,那么为了加强理解我们可以这样看:

3 N N N是2的左右子树,5 N N 6 N N是4的左右子树,2 4 是1 的左右子树,这样结合起来就好理解多了。

那么对于中序后序来说都是一样的,这里给代码,就不重复演示了:

void InOrder(TreeNode* root)
{
	if (root == NULL)
	{
		cout << "N ";
		return;
	}
	InOrder(root->left);
	cout << root->data << " ";
	InOrder(root->right);
}

void BackOrder(TreeNode* root)
{
	if (root == NULL)
	{
		cout << "N ";
		return;
	}
	BackOrder(root->left);
	BackOrder(root->right);
	cout << root->data << " ";
}

3 树的节点个数

树的节点个数问题,使用的是分而治之的思想,比如一个院,要统计有多少人,那么院长就发号司令,副院长去问班主任,班主任去问辅导员,辅导员去问班长,然后加上自己,最后就可以得到总总共的人数。

树的节点个数是一样的,求总节点个数,我们可以把树分为左右子树,把一个树拆分成无数的左右子树,统计每个左右子树的节点个数,相加即可。

代码实现:

size_t TreeSize(TreeNode* root)
{
	return root == NULL? 0 : 
		TreeSize(root->left) + TreeSize(root->right) + 1;
}

是的,代码就这么简单,这里给上递归展示图看看:

这里递归的是1的右子树。


4 树的高度

树的高度同理,我们可以理解为两个院的人比最高的,树的高度即我们同理,返回左右子树高度最高的即可,因为一个节点本身就算1,所以返回高度的时候需要加1,返回的条件就是节点为空,为空就返回0:

size_t TreeHeightError(TreeNode* root)
{
	return root == NULL ? 0 : 
		TreeHeight(root->left) > TreeHeight(root->right) ?
		TreeHeight(root->left) + 1 : TreeHeight(root->right) + 1;
}

以上代码看起来是没问题的,100个样例能跑过90的样例,但是数据量一多就会崩盘,因为这里没有记录左右子树的高度,也就是说会重复计算,这里不要小看重复计算,没有记录那么所有的计算都会翻二倍,而每一层都没有记录,所有越往层数多的走,就会变成2^n的重复计算,是指数量级的增长,所以我们应该记录数据:

size_t TreeHeight(TreeNode* root)
{
	if (root == NULL)	{
		return 0;
	}
	size_t leftHeight = TreeHeight(root->left);
	size_t rightHeight = TreeHeight(root->right);
	return leftHeight > rightHeight ? leftHeight + 1 : rightHeight + 1;
}

5 树的叶子节点个数

叶子节点即没有孩子节点的孩子,那么返回1的条件就是左右孩子都为空,如果不为空往下遍历就ok了:

size_t TreeLeaf(TreeNode* root)
{
	if (root == NULL)
	{
		return 0;
	}
	if (root->left == NULL && root->right == NULL)
	{
		return 1;
	}
	return TreeLeaf(root->left) + TreeLeaf(root->right);
}

6 树的第K层节点个数

这是本文最难的一个问题了,该问题的子问题是:

空节点的时候返回0,k = 1的时候返回1,K > 1的时候遍历即可,个人认为k - 1是这个问题的关键所在:

size_t TreeNodeK(TreeNode* root,size_t k)
{
	if (root == NULL)
	{
		return 0;
	}
	if (k == 1)
	{
		return 1;
	}
	return TreeNodeK(root->left, k - 1) +
		TreeNodeK(root->right, k - 1);
}

因为遍历是从根节点开始遍历的,所以减到k = 1的时候,是涉及到的那层,对该层进行判断,这是存在一个顺序问题,root == NULL应该在k == 1之前,因为递归到那一层,k = 1是铁定的,这时候就需要先判断是不是空节点,不是空节点再讨论K= 1的问题。

以上所有问题的测试代码:

int main()
{
	TreeNode* node = Tree();
	size_t k = 2;
	//前序遍历
	PrevOrder(node);
	cout << endl;
	//中序遍历
	InOrder(node);
	cout << endl;
	//后序遍历
	BackOrder(node);
	cout << endl;
	//树的节点个数
	cout << "The Tree's size is:" << 
		TreeSize(node) << endl;
	//树的高度
	cout << "The Tree's Height is:" <<
		TreeHeightError(node) << endl;
	//树的叶子节点个数
	cout << "The Tree's leaf is:" <<
		TreeLeaf(node) << endl;
	//树的第K层节点个数
	cout << "The Tree's leaf about K is:" <<
		TreeNodeK(node,k) << endl;

	return 0;
}

感谢阅读!

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

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

相关文章

三、安全工程练习题(CISSP)

1.三、安全工程练习题(CISSP)

找素数第二、三种方法

文章目录 第一种 &#xff1a;使用标签第二种&#xff1a;本质是方法的分装 第一种 &#xff1a;使用标签 没有使用信号量。break和continue作用范围只是最近的循环&#xff0c;无法控制外部循环。 此时使用标签 对外部循环进行操作。 package com.zhang; /* 找素数 第二种方…

【已解决】FileNotFoundError: [Errno 3] No such file or directory: ‘xxx‘

&#x1f60e; 作者介绍&#xff1a;我是程序员行者孙&#xff0c;一个热爱分享技术的制能工人。计算机本硕&#xff0c;人工制能研究生。公众号&#xff1a;AI Sun&#xff0c;视频号&#xff1a;AI-行者Sun &#x1f388; 本文专栏&#xff1a;本文收录于《AI实战中的各种bug…

【C语言】03.分支结构

本文用以介绍分支结构&#xff0c;主要的实现方式为if语句和switch语句。 一、if语句 1.1 if语句 if (表达式)语句表达式为真则执行语句&#xff0c;为假就不执行。在C语言中&#xff0c;0表示假&#xff0c;非0表示真.下图表示if的执行过程&#xff1a; 1.2 else语句 当…

数字孪生概念、数字孪生技术架构、数字孪生应用场景,深度长文学习

一、数字孪生起源与发展 1.1 数字孪生产生背景 数字孪生的概念最初由Grieves教授于2003年在美国密歇根大学的产品全生命周期管理课程上提出&#xff0c;并被定义为三维模型&#xff0c;包括实体产品、虚拟产品以及二者间的连接&#xff0c;如下图所示&#xff1a; 2011年&…

32位和64位的Windows7均不支持UEFI启动方式?试试看!

前言 今天小白突然想起&#xff1a;自己已经接近8年没有安装过32位的Windows系统了&#xff0c;这8年装的上百台电脑都是用的64位Windows。 今天 闲来无事 嗯……应该算是有小伙伴提出了个问题&#xff1a; 这位小伙伴表示&#xff1a;自己无论安装32位还是64位的Windows7都…

OSPF LSA头部详解

LSA概述 LSA是OSPF的本质 , 对于网工来说能否完成OSPF的排错就是基于OSPF的LSDB掌握程度 . 其中1/2类LAS是负责区域内部的 类似于设备的直连路由 . 加上对端的设备信息 3 类LSA是区域间的 指的是Area0和其他Area的区域间关系 , 设计多区域的初衷就是避免大型OSPF环境LSA太多…

14-特殊函数——静态函数、递归函数、函数指针、回调函数、内联函数、变参函数

14-特殊函数——静态函数、递归函数、函数指针、回调函数、内联函数、变参函数 文章目录 14-特殊函数——静态函数、递归函数、函数指针、回调函数、内联函数、变参函数一、静态函数1.1 语法 二、递归函数2.1 示例&#xff1a;输出n个自然数2.2 内存变化 三、函数指针四、指针函…

C++必修:探索C++的内存管理

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C学习 贝蒂的主页&#xff1a;Betty’s blog 1. C/C的内存分布 我们首先来看一段代码及其相关问题 int globalVar 1; static…

软件测试--Mysql快速入门

文章目录 软件测试-mysql快速入门sql主要划分mysql常用的数据类型sql基本操作常用字段的约束&#xff1a;连接查询mysql内置函数存储过程视图事务索引 软件测试-mysql快速入门 sql主要划分 sql语言主要分为&#xff1a; DQL&#xff1a;数据查询语言&#xff0c;用于对数据进…

作业-day-240607

思维导图 C编程 要求&#xff1a; 搭建一个货币的场景&#xff0c;创建一个名为 RMB 的类&#xff0c;该类具有整型私有成员变量 yuan&#xff08;元&#xff09;、jiao&#xff08;角&#xff09;和 fen&#xff08;分&#xff09;&#xff0c;并且具有以下功能&#xff1a;…

---java 抽象类 和 接口---

抽象类 再面向对对象的语言中&#xff0c;所以的对象都是通过类来描述的&#xff0c;但如果这个类无法准确的描述对象的 话&#xff0c;那么就可以把这个类设置为抽象类。 实例 这里用到abstract修饰&#xff0c;表示这个类或方法是抽象方法 因为会重写motifs里的show方法…

某药监局后缀(第一部分)

声明 本文章中所有内容仅供学习交流使用&#xff0c;不用于其他任何目的&#xff0c;抓包内容、敏感网址、数据接口等均已做脱敏处理&#xff0c;严禁用于商业用途和非法用途&#xff0c;否则由此产生的一切后果均与作者无关&#xff01; 本文章未经许可禁止转载&#xff…

微服务之熔断器

1、高并发带来的问题 在微服务架构中&#xff0c;我们将业务拆分成一个个的服务&#xff0c;服务与服务之间可以相互调用&#xff0c;但是由于网络原因 或者自身的原因&#xff0c;服务并不能保证服务的100%可用&#xff0c;如果单个服务出现问题&#xff0c;调用这个服务就会…

分别利用线性回归、多项式回归分析工资与年限的关系

一、线性回归&#xff1a; 实验思路&#xff1a; 先分析线性回归的代码&#xff0c;然后结合Salary_dataset.csv内容分析&#xff0c;编写代码。 实验代码&#xff1a; import pandas as pd import numpy as np from sklearn.linear_model import LinearRegression from skle…

【SQLAlChemy】filter过滤条件如何使用?

filter 过滤条件 生成 mock 数据 # 创建 session 对象 session sessionmaker(bindengine)()# 本地生成mock数据 for i in range(6):# 生成随机名字, 长度为4到7个字符name .join(random.choice(string.ascii_letters) for _ in range(random.randint(4, 7)))# 生成随机年龄…

C语言之常用字符串函数总结、使用和模拟实现

文章目录 目录 一、strlen 的使用和模拟实现 二、strcpy 的使用及模拟实现 三、strcat 的使用和模拟实现 四、strcmp 的使用和模拟实现 五、strncpy 的使用和模拟实现 六、strncat 的使用和模拟实现 七、strncmp 的使用和模拟实现 八、strstr 的使用和模拟实现 九、st…

Freeswitch-soundtouch-变声开发

文章目录 一、介绍二、安装soundtouch2.1 源码安装方式&#xff08;推荐&#xff09;2.1.1下载源码2.1.2解压2.1.3 编译2.1.4 迁移&#xff08;可选&#xff09; 2.2 apt-get 安装 三、使用3.1 终端使用3.2 Freeswitch使用3.2.1编译Freeswitch的mod_soundtouch3.2.2启用 mod_so…

如何秒杀系统架构设计

原文路径:https://learn.lianglianglee.com/%e4%b8%93%e6%a0%8f/%e5%a6%82%e4%bd%95%e8%ae%be%e8%ae%a1%e4%b8%80%e4%b8%aa%e7%a7%92%e6%9d%80%e7%b3%bb%e7%bb%9f/00%20%e5%bc%80%e7%af%87%e8%af%8d%20%e7%a7%92%e6%9d%80%e7%b3%bb%e7%bb%9f%e6%9e%b6%e6%9e%84%e8%ae%be%e8%ae%…

为什么选择海外服务器?

如何选择跨境电商服务器&#xff1a;详细指南 选择合适的服务器是跨境电商企业成功的基础。服务器的性能和稳定性直接影响着网站的访问速度、用户体验和安全性&#xff0c;进而影响着企业的销量和利润。那么&#xff0c;跨境电商企业该如何选择服务器呢&#xff1f; ​​​​​…