树低级(C语言版)

news2025/4/23 23:22:03

一.树基本计算规则

关于树的大部分知识点我们都讲过了,那么如果我给你树的节点,你可以算出叶子节点个数吗?

下面我们总结下一些计算规则:

1.父子计算规则:

parent=(child-1)/2;

leftchild=parent*2+1,rightchild=parent*2+2;

2.若规定根节点的层数为1,则一棵非空二叉树的第i层上最多有2^(i-1)个节点

3.深度为h的二叉树的最大结点数是2^h-1;

4. 对任何一棵二叉树, 如果度为0其叶结点个数为N0 , 度为2的分支结点个数为N2 ,则有

N0=N2+1
5. 若规定根节点的层数为1,具有n个结点的满二叉树的深度,h=log(n+1)  (log以2 为底,n+1为对数)
例题一:
1. 某二叉树共有 399 个结点,其中有 199 个度为 2 的结点,则该二叉树中的叶子结点数为
A 不存在这样的二叉树
B 200
C 198
D 199
规则4,结果为200个,即B
如果不知道规则,那么就可能浪费不必要的时间,大家可以自行去找其他题目。

二.层序遍历

上次我们留下层序遍历没实现,现在我们学习树更近一步了,我们可以实现层序遍历了,在实现前让我们来看看下面两个概念:
深度优先遍历(Depth First Search):简称DFS,前序遍历,中序遍历和后序遍历都是,即一种 递归的遍历方式从一个起点开始,沿着一条路径一直走到底,直到不能再走为止,然后回溯到上一个节点,继续遍历其他路径,直到所有节点都被遍历过。
广度优先遍历(Breadth First Search):又称为广度优先搜索,简称BFS。是一种分层的查找过程,每向前走一步可能访问一批顶点,不像深度优先搜索那样有往回退的情况,因此它不是一个递归的算法。它是一个逐层遍历的过程,层序遍历就是BFS。
大家可能都经历过这样一个过程,就是使用QQ时,会推荐好友的好友,这其实就是一个广度优先遍历的实例。
下面回归我们主题:
如何去写层序遍历呢?
这个时候,是不是可以想到用队列,如果我们将树根放进队列,如果我们出队列,将树根的左右子序列放入,这样一直循环,是不是就可以做到层序遍历。
下面请看代码:
// 层序遍历定义(一:一起遍历)
void LevelOrder(TreeNode* root)
{
	Queue s1;
	QueueInit(&s1);
	//判断是否为空,不空的话,先压进一个
	if(root)
		QueuePush(&s1, root);
	while (!QueueEmpty(&s1))
	{
		TreeNode* cur = QueueFront(&s1);
		QueuePop(&s1);
		printf("%d ", cur->date);
		//压入它的左右孩子
		if(root->left)
			QueuePush(&s1, root->left);
		if (root->right)
			QueuePush(&s1, root->right);
	}
	printf("\n");
}

还是以这个树,我们检查一下:
//动态开辟空间定义
TreeNode* BuyTreeNode(int x)
{
	//开辟树的空间
	TreeNode* node = (TreeNode*)malloc(sizeof(TreeNode));
	assert(node);
	node->date = x;
	node->left = NULL;
	node->right = NULL;
	return node;
}
//建立二叉树定义
TreeNode* CreateTree()
{
	//开辟树并且对每个进行赋值
	TreeNode* node1 = BuyTreeNode(1);
	TreeNode* node2 = BuyTreeNode(2);
	TreeNode* node3 = BuyTreeNode(3);
	TreeNode* node4 = BuyTreeNode(4);
	TreeNode* node5 = BuyTreeNode(5);
	TreeNode* node6 = BuyTreeNode(6);

	//确定指向
	node1->left = node2;
	node1->right = node4;
	node2->left = node3;
	node4->left = node5;
	node4->right = node6;

	return node1;
}
int main()
{
	TreeNode* root = CreateTree();
	LevelOrder(root);
	BinaryTreeDestory1(root);
	root = NULL;
	return 0;
}

结果:

是不是满足了层序遍历,好了,现在如果我让你将每层的结果,一行打印一层结果,该如何写呢?
// 层序遍历定义(二:逐层遍历)
void LevelOrder2(TreeNode* root)
{
	Queue s1;
	QueueInit(&s1);
	int size = 1;
	//判断是否为空,不空的话,先压进一个
	if (root != NULL)
		QueuePush(&s1, root);
	while (!QueueEmpty(&s1))
	{
		while (size>0)
		{
			TreeNode* cur = QueueFront(&s1);
			QueuePop(&s1);
			printf("%d ", cur->date);
			//压入它的左右孩子
			if (cur->left)
				QueuePush(&s1, cur->left);
			if (cur->right)
				QueuePush(&s1, cur->right);
			size--;
		}
		printf("\n");
		size = QueueSize(&s1);
	}
	printf("\n");
}

结果:(还是上面的树)

三.判断二叉树是否是完全二叉树

判断一棵树是不是完全二叉树,我们还是可以利用栈来实现,这里很好理解,就直接展示代码了,里面都有注释。
//判断二叉树是否是完全二叉树
bool BinaryTreeComplete(TreeNode* root)
{
	Queue s1;
	QueueInit(&s1);
	//判断是否为空,不空的话,先压进一个
	if (root != NULL)
		QueuePush(&s1, root);
	//找到第一个空
	while (!QueueEmpty(&s1))
	{
		
		TreeNode* cur = QueueFront(&s1);
		QueuePop(&s1);
		if (cur == NULL)
			break;
		//压入它的左右孩子,注意:不用判断了		
		QueuePush(&s1, cur->left);
		QueuePush(&s1, cur->right);		
	}
	//第二步:如果在栈中还有非空,就说明不是完全二叉树
	//定理:如果不是完全二叉树,那么栈里面此时一定有数据了
	while (!QueueEmpty(&s1))
	{
		TreeNode* cur2 = QueueFront(&s1);
		QueuePop(&s1);
		if (cur2 != NULL)
			return false;
	}
	return true;
}

检查:

//建立二叉树定义
TreeNode* CreateTree()
{
	//开辟树并且对每个进行赋值
	TreeNode* node1 = BuyTreeNode(1);
	TreeNode* node2 = BuyTreeNode(2);
	TreeNode* node3 = BuyTreeNode(3);
	TreeNode* node4 = BuyTreeNode(4);
	TreeNode* node5 = BuyTreeNode(5);
	TreeNode* node6 = BuyTreeNode(6);
	TreeNode* node7 = BuyTreeNode(7);

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

对于这棵树结果为:
当然,你也可以自己去检查我的结果
对于树的中等部分就补充到这里了,后面我们还会讲一些高级的树,元旦快乐,bye!!!

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

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

相关文章

swing快速入门(三十三)确认对话框

注释很详细,直接上代码 新增内容 1.确定对话框返回值对应值 2.为文本域增加滚动条 package swing31_40;import javax.swing.*; import java.awt.*; import java.awt.event.ActionEvent;public class swing_test_31 {// 创建一个JFrameJFrame jFrame new JFrame(…

《现代操作系统》第十二章习题答案

计算机硬件的改进主要归功于更小的晶体管。一些限制因素包括:(a) 光的波动性可能限制传统光刻技术制造集成电路的能力,(b) 固体中个别原子的迁移性可能导致非常薄的半导体、绝缘体和导体层的性能退化,(c) 背景辐射活性可能破坏分子键或影响非…

pytorch01:概念、张量操作、线性回归与逻辑回归

目录 一、pytorch介绍1.1pytorch简介1.2发展历史1.3pytorch优点 二、张量简介与创建2.1什么是张量?2.2Tensor与Variable2.3张量的创建2.3.1 直接创建torch.tensor()2.3.2 从numpy创建tensor 2.4根据数值创建2.4.1 torch.zeros()2.4.2 torch.zeros_like()2.4.3 torch…

回顾 2023,展望 2024

by zhengkai.blog.csdn.net 项目与心得 今年最大的项目和心得,非GCP莫属,作为全球顶尖的云平台, GCP有他的优势,也有很多难用的地方。但是作为当时的一个strategic solution,我们的印度本地化项目必须使用GCP&#xf…

Linux权限的基本理解

一:🚩Linux中的用户 1.1🥦用户的分类 🌟在Linux中用户可以被分为两种用户: 超级用户(root):可以在Linux系统中做各种事情而不被约束普通用户:只能做有限的事情被权限约束 在实际操作时超级用户的命令提示符为#,普通用户的命令提示符为$,可…

数模混合SoC芯片中LEF2Milkyway的golden flow

在数模混合芯片中的项目中,特别是数字模块很少甚至只有一个简单的数字控制逻辑时,我们要做数字模块的后端实现时,通常模拟那边会问我们实现需要他们提供哪些数据。 通常来说,我们可以让模拟设计提供数字模块的GDS或LEF文件即可。…

nodejs+vue+微信小程序+python+PHP特困救助供养信息管理系统-计算机毕业设计推荐

通过走访某特困救助供养机构实际情况,整理特困救助供养机构管理的业务流程,分析当前特困救助供养机构管理存在的各种问题,利用软件开发思想对特困救助供养机构特困救助供养机构管理进行系统设计分析。通过服务端程序框架进行设计,…

网际协议IPv4

基本介绍 网际协议IP是TCP/IP体系中两个重要的协议之一。IPv4虽有最终被IPv6取代的趋势,但它仍是当前使用的最重要的因特网协议。 与IP配套使用的还有3个协议: 地址解析协议ARP(Address Resolution Protocol)因特网控制报文协议ICMP(Internet Control …

Docker 入门 ------容器互通以及Dockerfile

1. 端口映射以及容器互联 Docker 除了通过网络访问,还提供了两种很方便的功能来满足服务访问的基本需求: 允许映射容器内应用的服务端口到本地宿主主机互联机制实现多个容器间通过容器名来快速访问 1.1 容器映射实现访问容器 1.1.1 从外部访问容器应…

中间人攻击是什么,会产生哪些危害,如何有效防止中间人攻击

简介 中间人攻击(Man-in-the-Middle Attack,简称MITM攻击)是一种网络攻击,其原理是攻击者通过各种技术手段将受攻击者控制的一台计算机虚拟放置在网络连接中的两台通信计算机之间,这台计算机称为“中间人”。在攻击过…

【滑动窗口】C++算法:可见点的最大数目

作者推荐 动态规划 多源路径 字典树 LeetCode2977:转换字符串的最小成本 本题涉及知识点 滑动窗口 LeetCode 1610可见点的最大数目 给你一个点数组 points 和一个表示角度的整数 angle ,你的位置是 location ,其中 location [posx, posy] 且 point…

Linux文件系统结构及相关命令1(man pwd ls ctrl +Shift +T ls /etc)

Linux的文件系统结构 某所大学的学生可能在一两万人左右,通常将学生分配在以学院-系班为单位的分层组织机构中。 如何查找一名学生? 最笨的办法:依次问询大学中的每一个学生,直到找到为止。 查询效率高的方法:按照从…

2023年新一代开发者工具 Vue ,正式开源!

以下文章来源于前端充电宝 ,作者CUGGZ 近日,Vue 新一代开发者工具(DevTools)正式开源!Vue DevTools 是一个旨在增强 Vue 开发人员体验的工具,它提供了一些功能来帮助开发者更好地了解 Vue 应用。下面就来看…

用IDEA创建/同步到gitee(码云)远程仓库(保姆级详细)

前言: 笔者最近在学习java,最开始在用很笨的方法:先克隆远程仓库到本地,再把自己练习的代码从本地仓库上传到远程仓库,很是繁琐。后发现可以IDEA只需要做些操作可以直接把代码上传到远程仓库,也在网上搜了些…

【FileZilla】的基本使用

一、FileZilla的使用 1.1 FileZilla简介 1.2 软件下载 到官方网站下载 FileZilla 的服务端和客户端程序 FileZilla - The free FTP solution 自行下载即可 1.3 软件安装 (1)先安装服务端【傻瓜式安装】,一直下一步下一步安装即可 &#xf…

Python绘制高级图表(1):绘制条形统计图

一、初始化 1. 引入库,设置画笔 from turtle import * t Turtle() t.color("black") t.width(3)2. 为了美观,画出xy轴 (1) 普通型 from turtle import * t Turtle() t.color("black") t.width(3)# 以画布为600 * 600为例 # 1.…

搭建maven私服

maven maven简介 什么是maven? Maven这个单词来自于意第绪语(犹太语),意为知识的积累。 Maven项目对象模型(POM),可以通过一小段描述信息来管理项目的构建,报告和文档的项目管理工具软件。 Maven 除了以…

算法训练day53|动态规划part14

参考:代码随想录 1143.最长公共子序列 重点:状态的转移与递推公式的确定 本题和动态规划:718. 最长重复子数组 (opens new window)区别在于这里不要求是连续的了,但要有相对顺序,即:"ace" 是 …

Spark任务调度与数据本地性

Apache Spark是一个分布式计算框架,用于处理大规模数据。了解Spark任务调度与数据本地性是构建高效分布式应用程序的关键。本文将深入探讨Spark任务调度的流程、数据本地性的重要性,并提供丰富的示例代码来帮助大家更好地理解这些概念。 Spark任务调度的…

小程序面试题 | 17.精选小程序面试题

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云…