数据结构:二叉树
数据结构:二叉树
文章目录
- 数据结构:二叉树
- 1.一些特殊的二叉树
- 1.满二叉树
- 2.完全二叉树
- 2.手动创建一颗二叉树
- 3.二叉树深度优先遍历
- 4.二叉树层序遍历
- 5.二叉树基础操作
- 1.创建二叉树
- 2.二叉树节点个数
- 3.二叉树叶子节点个数
- 4.二叉树的高度
- 5.二叉树第k层节点个数
- 6.二叉树查找值为x的节点
- 7.层序遍历
- 8.二叉树销毁
- 9.判断二叉树是否是完全二叉树
二叉树
1.一些特殊的二叉树
1.满二叉树
满二叉树:一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是
说,如果==一个二叉树的层数为K,且结点总数是2^k-1== ,则它就是满二叉树。
2.完全二叉树
完全二叉树:完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K
的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从1至n的结点一 一对
应时称之为完全二叉树。 **要注意的是满二叉树是一种特殊的完全二叉树**。
2.手动创建一颗二叉树
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>// 手动快速创建一棵简单的二叉树来测试三种深度优先遍历方式(前中后序遍历)(后续学习递归构建二叉树才是真正常用的方法)
typedef int BinaryTreeDataType;typedef struct BinaryTreeNode
{BinaryTreeDataType val;struct BinaryTreeNode* left;struct BinaryTreeNode* right;
} BTNode;BTNode* CreateBTNode(BinaryTreeDataType x)
{BTNode* NewBTNode = (BTNode*)malloc(sizeof(BTNode));if (NewBTNode == NULL){perror("malloc fail");exit(-1);}NewBTNode->val = x;NewBTNode->left = NULL;NewBTNode->right = NULL;return NewBTNode;
}
BTNode* CreatBinaryTree()
{BTNode* root = CreateBTNode(1);BTNode* node2 = CreateBTNode(2);BTNode* node3 = CreateBTNode(3);BTNode* node4 = CreateBTNode(4);BTNode* node5 = CreateBTNode(5);BTNode* node6 = CreateBTNode(6);root->left = node2;root->right = node4;node2->left = node3;node4->left = node5;node4->right = node6;return root;
}
3.二叉树深度优先遍历
前序、中序和后序遍历都属于「深度优先遍历 depth-first traversal, DFS」,它体现了一种“先走到尽头,再回溯继续”的遍历方式。
// 前序遍历
void Preorder(BTNode* root)
{if (root == NULL){printf("N ");return;}printf("%d ", root->val);Preorder(root->left);Preorder(root->right);
}// 中序遍历
void Inorder(BTNode* root)
{if (root == NULL){printf("N ");return;}Inorder(root->left);printf("%d ", root->val);Inorder(root->right);
}
// 后序遍历
void Postorder(BTNode* root)
{if (root == NULL){printf("N ");return;}Postorder(root->left);Postorder(root->right);printf("%d ", root->val);
}
4.二叉树层序遍历
层序遍历本质上属于「广度优先遍历 breadth-first traversal, BFS」,它体现了一种“一圈一圈向外扩展”的逐层遍历方式。
利用队列先进先出的性质:父亲先进队列,父亲出来时再带孩子进队列
void BinaryTreeLevelOrder(BTNode* root)
{// 利用队列 实现二叉树的层序遍历 (队列先进先出性质)Queue* queuehead = Init();Push(&queuehead, root);while (!Empty(queuehead)){// 遍历二叉树入队列并挨个打印值BTNode* front = Peek(&queuehead);printf("%c ", front->data);Pop(&queuehead);if (front->left != NULL){Push(&queuehead, front->left);}if (front->right != NULL){Push(&queuehead, front->right);}}// 销毁队列Destroy(&queuehead);
}
5.二叉树基础操作
1.创建二叉树
typedef char BTDataType;
typedef struct BinaryTreeNode
{BTDataType data;struct BinaryTreeNode* left;struct BinaryTreeNode* right;
}BTNode;
// 通过 前序遍历 的数组"ABD##E#H##CF##G##"构建二叉树BTNode* BinaryTreeCreate(BTDataType* parray, int* pi)
{// 前序遍历创建二叉树if (parray[*pi] == '#'){(*pi)++;return NULL;}BTNode* root = (BTNode*)malloc(sizeof(BTNode));if (root == NULL){perror("malloc fail");exit(-1);}root->data = parray[(*pi)++];root->left = BinaryTreeCreate(parray, pi);root->right = BinaryTreeCreate(parray, pi);return root;
}
2.二叉树节点个数
/*
写法一:遍历计数
*/
int BinaryTreeSize(BTNode* root)
{// 遍历二叉树计算节点个数static int size = 0;// 函数中使用静态不能完全解决问题,无法处理多次计算的情况if (root == NULL){return 0;}size++;BinaryTreeSize(root->left);BinaryTreeSize(root->right);return size;
}/*
写法二:递归分治子问题
*/
int BinaryTreeSize(BTNode* root)
{return root == NULL ? 0 : BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1;//节点个数 = 左子树节点个数 + 右子树节点个数 + 1
}
3.二叉树叶子节点个数
int BinaryTreeLeafSize(BTNode* root)
{if (root == NULL){return 0;}// 叶子节点个数 = 左子树叶子节点个数 + 右子树叶子节点个数return root->left == NULL && root->right == NULL ? 1 : BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}
4.二叉树的高度
/*
写法1:
*/
int BinaryTreeHight(BTNode* root)
{if (root == NULL){return 0;}// 此处递归分治多次重复导致效率过低(原因是此逻辑中比较时候进行了递归分治,计算的时候又重复进行了计算)return BinaryTreeHight(root->left) > BinaryTreeHight(root->right)|| BinaryTreeHight(root->left) == BinaryTreeHight(root->right)? BinaryTreeHight(root->left) + 1: BinaryTreeHight(root->right) + 1;// 二叉树的高度 = 左子树与右子树相比,高度更高的那棵树的高度 + 1
}/*
写法2:
*/
int BinaryTreeHight(BTNode* root)
{if (root == NULL){return 0;}//提前记录高度int lefthight = BinaryTreeHight(root->left);int righthight = BinaryTreeHight(root->right);// 二叉树的高度 = 左子树与右子树相比,高度更高的那棵树的高度 + 1return lefthight > righthight|| lefthight == righthight? lefthight + 1: righthight + 1;}
5.二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k)
{if (root == NULL){return 0;}if (root != NULL && k == 1){return 1;}// 问题拆分:第k层节点个数 = 左子树第k-1层节点数 + 右子树第k-层节点数return BinaryTreeLevelKSize(root->left, k - 1) + BinaryTreeLevelKSize(root->right, k - 1);}
6.二叉树查找值为x的节点
BTNode* BinaryTreeFind(BTNode* root, BTDataType x)
{if (root == NULL){return NULL;}// 前序遍历二叉树寻找该节点if (root->data == x){return root;}else{BTNode* leftnode = BinaryTreeFind(root->left, x);if (leftnode){return leftnode;}BTNode* rightnode = BinaryTreeFind(root->right, x);if (rightnode){return rightnode;}//如果左右节点都不是我们要找的该节点则返回空if (leftnode == NULL && rightnode == NULL){return NULL;}}
}
7.层序遍历
void BinaryTreeLevelOrder(BTNode* root)
{// 利用队列 实现二叉树的层序遍历 (队列先进先出性质)Queue* queuehead = Init();Push(&queuehead, root);while (!Empty(queuehead)){// 遍历二叉树入队列并挨个打印值BTNode* front = Peek(&queuehead);printf("%c ", front->data);Pop(&queuehead);if (front->left != NULL){Push(&queuehead, front->left);}if (front->right != NULL){Push(&queuehead, front->right);}}// 销毁队列Destroy(&queuehead);
}
变形:如何控制一层一层打印并换行?
// 层序遍历变形换行打印
void BinaryTreeLevelOrder(BTNode* root)
{// 利用队列 实现二叉树的层序遍历 (队列先进先出性质)Queue* queuehead = Init();Push(&queuehead, root);// 根据每一层的数据的个数得出打印多少次后进行一次换行int levelsize = 1;while (!Empty(queuehead)){// 一层一层出while (levelsize--){// 遍历二叉树入队列BTNode* front = Peek(&queuehead);printf("%c ", front->data);Pop(&queuehead);if (front->left != NULL){Push(&queuehead, front->left);}if (front->right != NULL){Push(&queuehead, front->right);}}printf("\n");levelsize = Size(queuehead);}// 销毁队列Destroy(&queuehead);
}
8.二叉树销毁
// 二叉树销毁
void BinaryTreeDestory(BTNode* root)
{// 走后序遍历更方便销毁if (root == NULL){return;}BinaryTreeDestory(root->left);BinaryTreeDestory(root->right);free(root);
}
9.判断二叉树是否是完全二叉树
思路:完全二叉树只有最后一层会出现 NULL 值,而且出现了 NULL 值,则后面不会再出现非空的值。我们可以通过层序遍历的思想,一层一层遍历,如果遇到空结点,记录一下,然后继续遍历,若是后面出现了非空值,则说明该二叉树不是完全二叉树。
// 判断二叉树是否是完全二叉树
// 参数:root,一个指向二叉树根节点的指针
bool BinaryTreeComplete(BTNode* root)
{Queue* queuehead = Init();// 将二叉树的根节点入队列Push(&queuehead, root);while (!Empty(queuehead)){BTNode* front = Peek(&queuehead);if (front == NULL){break;}Pop(&queuehead);Push(&queuehead, front->left);Push(&queuehead, front->right);}// 当第一次遍历的时候出道空代表应当结束了,第二次继续往后遍历如果队列中还有非空代表不是完全二叉树while (!Empty(queuehead)){BTNode* front = Peek(&queuehead);Pop(&queuehead);if (front != NULL){return false;}}return true;
}
相关文章:

数据结构:二叉树
数据结构:二叉树 文章目录 数据结构:二叉树1.一些特殊的二叉树1.满二叉树2.完全二叉树 2.手动创建一颗二叉树3.二叉树深度优先遍历4.二叉树层序遍历5.二叉树基础操作1.创建二叉树2.二叉树节点个数3.二叉树叶子节点个数4.二叉树的高度5.二叉树第k层节点个…...

HTTP超文本传输协议
原文链接: 1.5 万字 40 张图解 HTTP 常见面试题(值得收藏)_图解http 小林-CSDN博客https://blog.csdn.net/qq_34827674/article/details/124089736?ops_request_misc%257B%2522request%255Fid%2522%253A%2522170521531616777224478386%252…...

视频SDK的技术架构优势和价值
为了满足企业对于高质量视频的需求,美摄科技推出了一款强大的视频SDK(软件开发工具包),旨在帮助企业轻松实现高效、稳定的视频功能,提升用户体验,增强企业竞争力。 一、美摄视频SDK的技术实现方式 美摄视…...

Invalid bound statement (not found)(xml文件创建问题)
目录 解决方法: 这边大致讲一下我的经历,不想看的直接点目录去解决方法 今天照着老师视频学习,中间老师在使用动态SQL时,直接复制了一份,我想这么简单的一个,我直接从网上找内容创建一个好了,…...
正则表达式2 常见模式
继上次的正则表达式速攻1/2-CSDN博客 还有一些常见的匹配模式可以直接使用 电子邮箱 xxxxxx.域名 的情况 \b[A-Za-z0-9._%-][A-Za-z0-9.-]\.[A-Z|a-z]{2,}\bhttp或者https网址 的情况 http[s]?://(?:[a-zA-Z]|[0-9]|[$-_.&]|[!*\\(\\),]|(?:%[0-9a-fA-F][0-9a-fA-F…...

前端对接电子秤、扫码枪设备serialPort 串口使用教程
因为最近工作项目中用到了电子秤,需要对接电子秤设备。以前也没有对接过这种设备,当时也是一脸懵逼,脑袋空空。后来就去网上搜了一下前端怎么对接,然后就发现了SerialPort串口。 Serialport 官网地址:https://serialpo…...
LeeCode前端算法基础100题(18)整数转罗马数字
一、问题详情: 罗马数字包含以下七种字符: I, V, X, L,C,D 和 M。 字符 数值 I 1 V 5 X 10 L 50 C 100 D 500 M 1000 例如, 罗马数字 2 写做 II ,即为两个并列的 1…...

【C++ 程序设计入门基础】- 第4节-函数
1、函数 函数是对实现某一功能的代码的模块化封装。 函数的定义: 标准函数: 输入 n 对整数的 a、b ,输出它们的和。 #include <iostream> #include <windows.h> using namespace std;int add(int a,int b);//函数原型声明int…...

华为数通HCIA题库(750题)
完整题库在这里:华为数通HCIA-RS题库注释版-加水印.pdf资源-CSDN文库 此处只节选几题。 1.网络管理员在网络中捕获到了一个数据帧,其目的MAC地址是01-00-5E-AO-B1-C3。关于该MAC地址的说法正确的是( )。 A.它是一个单播MAC地址 B.它是一个广播…...

SpringIOC之support模块GenericXmlApplicationContext
博主介绍:✌全网粉丝5W,全栈开发工程师,从事多年软件开发,在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战,博主也曾写过优秀论文,查重率极低,在这方面有丰富的经验…...
CCF认证+蓝桥杯习题训练
贪心 *上取整公式* *代码展示* #include <iostream> #include <cstring> #include <algorithm>using namespace std;const int N 1e5 10;typedef long long LL;int v[N] , a[N];int main() {int n , d;cin >> n >> d;for(int i 1 ; i < n…...

vue前端开发自学基础,动态切换组件的显示
vue前端开发自学基础,动态切换组件的显示!这个是需要借助于,一个官方提供的标签,名字叫【Component】-[代码demo:<component :is"ComponetShow"></component>]。 下面看看代码详情。 <template><h3>动态…...

16.桥接模式
桥接模式 介绍 桥接模式是一种结构型设计模式,它通过将抽象部分与实现部分分离,使它们可以独立变化。这种模式通过组合的方式来实现,而不是继承。桥接模式通过将抽象和实现解耦,从而实现抽象和实现的分离,使得系统更加…...

【网络安全】【密码学】【北京航空航天大学】实验一、数论基础(上)【C语言和Java实现】
实验一、数论基础(上) 一、实验目的 1、通过本次实验,熟悉相关的编程环境,为后续的实验做好铺垫; 2、回顾数论学科中的重要基本算法,并加深对其的理解,为本学期密码学理论及实验课程打下良好…...
Go语言的sync.Pool如何使用?使用场景具体有哪些?
sync.Pool 是 Go 标准库中提供的一个对象池(Object Pool)的实现。对象池是一种用于缓存和复用对象的机制,可以在一定程度上减轻内存分配的开销。sync.Pool 专门用于管理临时对象,适用于一些需要频繁创建和销毁的短暂对象ÿ…...

MySQL单表查询练习题
一、创建表的素材 表名:worker——表中字段均为中文,比如:部门号、工资、职工号、参加工作等 CREATE TABLE worker ( 部门号 int(11) NOT NULL, 职工号 int(11) NOT NULL, 工作时间 date NOT NULL, 工资 float(8,2) NOT NULL, 政治面貌 …...
Spring MVC中@Controller和@RestController的区别
Controller 和 RestController 是 Spring MVC 中用于处理 HTTP 请求的注解,它们有以下区别: 返回值处理方式: Controller 用于定义一个传统的 Spring MVC 控制器,它的方法通常返回视图名称或 ModelAndView 对象,由视图…...

Flink定制化功能开发,demo代码
前言: 这是一个Flink自定义开发的基础教学。本文将通过flink的DataStream模块API,以kafka为数据源,构建一个基础测试环境;包含一个kafka生产者线程工具,一个自定义FilterFunction算子,一个自定义MapFunctio…...

Edge浏览器入门
关于作者: CSDN内容合伙人、技术专家, 从零开始做日活千万级APP,带领团队单日营收超千万。 专注于分享各领域原创系列文章 ,擅长java后端、移动开发、商业化变现、人工智能等,希望大家多多支持。 目录 一、导读二、概览…...
Go语言的调度器
简介 Go语言的调度器是一个非常强大的工具,它可以帮助我们轻松地实现并发编程。调度器的工作原理是将多个协程映射到多个操作系统线程上,并根据协程的状态来决定哪个协程应该在哪个线程上运行。 调度器有两种主要策略: 协作式调度…...

大数据学习栈记——Neo4j的安装与使用
本文介绍图数据库Neofj的安装与使用,操作系统:Ubuntu24.04,Neofj版本:2025.04.0。 Apt安装 Neofj可以进行官网安装:Neo4j Deployment Center - Graph Database & Analytics 我这里安装是添加软件源的方法 最新版…...

【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...

Vue3 + Element Plus + TypeScript中el-transfer穿梭框组件使用详解及示例
使用详解 Element Plus 的 el-transfer 组件是一个强大的穿梭框组件,常用于在两个集合之间进行数据转移,如权限分配、数据选择等场景。下面我将详细介绍其用法并提供一个完整示例。 核心特性与用法 基本属性 v-model:绑定右侧列表的值&…...
如何为服务器生成TLS证书
TLS(Transport Layer Security)证书是确保网络通信安全的重要手段,它通过加密技术保护传输的数据不被窃听和篡改。在服务器上配置TLS证书,可以使用户通过HTTPS协议安全地访问您的网站。本文将详细介绍如何在服务器上生成一个TLS证…...

安宝特案例丨Vuzix AR智能眼镜集成专业软件,助力卢森堡医院药房转型,赢得辉瑞创新奖
在Vuzix M400 AR智能眼镜的助力下,卢森堡罗伯特舒曼医院(the Robert Schuman Hospitals, HRS)凭借在无菌制剂生产流程中引入增强现实技术(AR)创新项目,荣获了2024年6月7日由卢森堡医院药剂师协会࿰…...
Kafka主题运维全指南:从基础配置到故障处理
#作者:张桐瑞 文章目录 主题日常管理1. 修改主题分区。2. 修改主题级别参数。3. 变更副本数。4. 修改主题限速。5.主题分区迁移。6. 常见主题错误处理常见错误1:主题删除失败。常见错误2:__consumer_offsets占用太多的磁盘。 主题日常管理 …...

第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10+pip3.10)
第一篇:Liunx环境下搭建PaddlePaddle 3.0基础环境(Liunx Centos8.5安装Python3.10pip3.10) 一:前言二:安装编译依赖二:安装Python3.10三:安装PIP3.10四:安装Paddlepaddle基础框架4.1…...
Python爬虫实战:研究Restkit库相关技术
1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的有价值数据。如何高效地采集这些数据并将其应用于实际业务中,成为了许多企业和开发者关注的焦点。网络爬虫技术作为一种自动化的数据采集工具,可以帮助我们从网页中提取所需的信息。而 RESTful API …...
手动给中文分词和 直接用神经网络RNN做有什么区别
手动分词和基于神经网络(如 RNN)的自动分词在原理、实现方式和效果上有显著差异,以下是核心对比: 1. 实现原理对比 对比维度手动分词(规则 / 词典驱动)神经网络 RNN 分词(数据驱动)…...

LINUX编译vlc
下载 VideoLAN / VLC GitLab 选择最新的发布版本 准备 sudo apt install -y xcb bison sudo apt install -y autopoint sudo apt install -y autoconf automake libtool编译ffmpeg LINUX FFMPEG编译汇总(最简化)_底部的附件列表中】: ffmpeg - lzip…...