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

数据结构之二叉树详解:从原理到实现

1. 什么是二叉树?

二叉树(Binary Tree)是一种树形数据结构,其中每个节点最多有两个子节点,分别被称为左子节点右子节点。二叉树可以用来表示层次关系,如文件目录组织结构,或用于快速查找、排序和决策问题。其结构如下:

2. 二叉树的基本术语

在了解二叉树之前,我们需要掌握一些关键术语:

  • 节点(Node):二叉树的基本单元,包含数据和指向子节点的指针。
  • 根节点(Root):二叉树的最顶端节点。
  • 叶子节点(Leaf):没有子节点的节点。
  • 高度(Height):从某节点到叶子节点的最长路径上的边数。
  • 深度(Depth):从根节点到某节点所经过的边数。
  • 子树(Subtree):由某节点及其子节点组成的部分树。

3. 二叉树的分类

根据节点的分布特点,二叉树有多种类型:

  1. 满二叉树:每个节点都有两个子节点,且所有叶子节点在同一层。
  2. 完全二叉树:除了最后一层外,其他层的节点都被填满,最后一层的叶子节点从左到右连续排列。
  3. 二叉搜索树(BST):对于任意一个节点,左子树中的所有节点值都小于该节点值,右子树中的所有节点值都大于该节点值。
  4. 平衡二叉树:左右子树的高度差不超过1。
4. 二叉树的存储结构
二叉树可以通过两种方式存储:
  1. 链式存储:用链表表示,每个节点包含数据和左右子节点的指针。
  2. 顺序存储:用数组表示,通常用于完全二叉树或满二叉树。
链式存储

在C语言中,链式存储通常使用结构体来定义二叉树节点:

​
#include <stdio.h>
#include <stdlib.h>// 定义二叉树节点
struct TreeNode {int data;                // 数据域struct TreeNode* left;   // 左子节点指针struct TreeNode* right;  // 右子节点指针
};// 创建新节点
struct TreeNode* createNode(int data) {struct TreeNode* newNode = (struct TreeNode*)malloc(sizeof(struct TreeNode));newNode->data = data;newNode->left = NULL;newNode->right = NULL;return newNode;
}​
顺序存储

顺序存储常用数组表示,用于完整或接近完整的二叉树。节点的存储规则如下:

  • 根节点存储在索引1(或0)。
  • 索引为i的节点的左子节点在2*i位置,右子节点在2*i+1位置。
  • 父节点在索引i/2位置。
5. 二叉树的基本操作
1. 插入节点

插入节点的方式取决于二叉树的类型。在二叉搜索树(BST)中,插入节点时需要保持左小右大的规则。

​
struct TreeNode* insert(struct TreeNode* root, int data) {if (root == NULL) {return createNode(data);  // 如果当前节点为空,创建新节点}if (data < root->data) {root->left = insert(root->left, data);  // 插入左子树} else if (data > root->data) {root->right = insert(root->right, data); // 插入右子树}return root;
}​
2. 查找节点

在二叉搜索树中查找节点时,可以利用其有序性快速定位目标节点。

​
struct TreeNode* search(struct TreeNode* root, int key) {if (root == NULL || root->data == key) {return root;  // 找到节点或到达空节点}if (key < root->data) {return search(root->left, key);  // 在左子树中查找} else {return search(root->right, key); // 在右子树中查找}
}​
3. 遍历二叉树

二叉树的遍历方式分为以下几种:

  • 前序遍历(Pre-order):根节点 -> 左子树 -> 右子树
  • 中序遍历(In-order):左子树 -> 根节点 -> 右子树
  • 后序遍历(Post-order):左子树 -> 右子树 -> 根节点
  • 层序遍历(Level-order):按层次从上到下逐层遍历
前序遍历
​
void preOrder(struct TreeNode* root) {if (root == NULL) return;printf("%d ", root->data);preOrder(root->left);preOrder(root->right);
}​
中序遍历
​
void inOrder(struct TreeNode* root) {if (root == NULL) return;inOrder(root->left);printf("%d ", root->data);inOrder(root->right);
}​
后序遍历
​
void postOrder(struct TreeNode* root) {if (root == NULL) return;postOrder(root->left);postOrder(root->right);printf("%d ", root->data);
}​
4. 删除节点

在二叉搜索树中删除节点我们需要考虑三种情况

  1. 被删除节点是叶子节点。
  2. 被删除节点有一个子节点。
  3. 被删除节点有两个子节点。
​
struct TreeNode* deleteNode(struct TreeNode* root, int key) {if (root == NULL) return root;if (key < root->data) {root->left = deleteNode(root->left, key);  // 在左子树中删除} else if (key > root->data) {root->right = deleteNode(root->right, key); // 在右子树中删除} else {// 找到要删除的节点if (root->left == NULL) {struct TreeNode* temp = root->right;free(root);return temp;} else if (root->right == NULL) {struct TreeNode* temp = root->left;free(root);return temp;}// 有两个子节点,找右子树的最小节点struct TreeNode* temp = root->right;while (temp->left != NULL) {temp = temp->left;}root->data = temp->data; // 用最小值替换当前节点root->right = deleteNode(root->right, temp->data); // 删除最小节点}return root;
}​
6. 二叉树的应用
  1. 二叉搜索树(BST):用于快速查找、插入和删除操作。
  2. 堆(Heap):用于优先队列和排序。
  3. 表达式树:用于表示算术表达式。
  4. Huffman树:用于数据压缩。
  5. 平衡二叉树(AVL/红黑树):用于高效的动态数据结构

表示文件系统的目录树结构

相关文章:

数据结构之二叉树详解:从原理到实现

1. 什么是二叉树&#xff1f; 二叉树&#xff08;Binary Tree&#xff09;是一种树形数据结构&#xff0c;其中每个节点最多有两个子节点&#xff0c;分别被称为左子节点和右子节点。二叉树可以用来表示层次关系&#xff0c;如文件目录、组织结构&#xff0c;或用于快速查找、…...

iOS 系统中使用 webView 打印 html 的打印边距问题

需求是使用系统提供的打印功能将HTML代码打印出来 1、使用CSS page 设置边距&#xff08;iOS不生效&#xff09; page {margin: 0;padding: 0;size: A6 portrait; }在 Android 中边距设置生效的&#xff0c;但是在 iOS 系统使用CSS page规则是不生效的 当从 iOS 系统打印网页…...

如何在ubuntu上调试core dump

启用core dump 确认ulimit 状态 ulimit -c 如果输出是0&#xff0c;表示core dump被禁用了 运行 ulimit -c unlimited 再次运行 ulimit -c 确认输出是ulimited 设置core dump路径和文件名格式 下面命令表示设置core dump文件在当前目录&#xff08;%e表示程序名&#x…...

基于 JNI + Rust 实现一种高性能 Excel 导出方案(上篇)

每个不曾起舞的日子&#xff0c;都是对生命的辜负。 ——尼采 一、背景&#xff1a;Web 导出 Excel 的场景 Web 导出 Excel 功能在数据处理、分析和共享方面提供了极大的便利&#xff0c;是许多 Web 应用程序中的重要功能。以下是一些典型的场景&#xff1a; 数据报表导出&am…...

【Maven】依赖管理

4. Maven的依赖管理 在 Java 开发中&#xff0c;项目的依赖管理是一项重要任务。通过合理管理项目的依赖关系&#xff0c;我们可以有效的管理第三方库&#xff0c;模块的引用及版本控制。而 Maven 作为一个强大的构建工具和依赖管理工具&#xff0c;为我们提供了便捷的方式来管…...

springboot/ssm高校超市管理系统Java商品出入库供应商管理系统web源码wms

springboot/ssm高校超市管理系统Java商品出入库供应商管理系统web源码wms 基于springboot(可改ssm)vue项目 开发语言&#xff1a;Java 框架&#xff1a;springboot/可改ssm vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&a…...

小程序-基于java+SpringBoot+Vue的微信小程序养老院系统设计与实现

项目运行 1.运行环境&#xff1a;最好是java jdk 1.8&#xff0c;我们在这个平台上运行的。其他版本理论上也可以。 2.IDE环境&#xff1a;IDEA&#xff0c;Eclipse,Myeclipse都可以。推荐IDEA; 3.tomcat环境&#xff1a;Tomcat 7.x,8.x,9.x版本均可 4.硬件环境&#xff1a…...

宠物电商对接美团闪购:实现快速配送与用户增值

随着宠物行业的快速发展&#xff0c;宠物电商市场也在不断扩张。消费者的需求不再局限于传统的线上购物模式&#xff0c;越来越多的人开始追求更快捷的配送服务和更优质的购物体验。为了适应这一趋势&#xff0c;许多宠物电商平台开始寻求与本地配送平台合作&#xff0c;以提供…...

Vue中使用<Transition>与<TransitionGroup>

目录 介绍 CSS过渡类 为过渡效果命名 CSS的transition CSS的transform 性能考量 出现时过渡 元素间过渡 过渡模式 使用Key属性过渡 和的区别 进入/离开动画 移动动画 一个购物车飞跃例子 介绍 传统HTML中&#xff0c;我们可以使用CSS属性&#xff1a;“animation”…...

Algorithms and Data Structures in C++ by Mohammed Yasir Eramangadan

MP4 创建 |视频&#xff1a;h264、1280720 |音频&#xff1a;AAC&#xff0c;44.1 KHz&#xff0c;2 通道 类型&#xff1a;在线学习 |语言&#xff1a;英文 字幕 |持续时间&#xff1a; 159 讲座 &#xff08; 10h 43m &#xff09; |大小&#xff1a; 3.5 GB “通过专家制作…...

2024广东省职业技能大赛云计算——构建CICD 部署2048小游戏

构建CI/CD 前言 题目如下&#xff1a; 构建CI/CD 编写流水线脚本.gitlab-ci.yml触发自动构建&#xff0c;具体要求如下&#xff1a; &#xff08;1&#xff09;基于镜像maven:3.6-jdk-8构建项目的drone分支&#xff1b; &#xff08;2&#xff09;构建镜像的名称&#xff1a…...

React 条件渲染

React 条件渲染 React 条件渲染是一种在 React 应用程序中根据不同的条件显示不同组件或内容的技巧。它是 React 响应用户输入、状态变化或数据变化的核心机制之一。本文将深入探讨 React 条件渲染的概念、用法和最佳实践。 目录 条件渲染的基本概念使用 JavaScript 运算符进…...

Hadoop生态圈框架部署(九)- Hive部署

文章目录 前言一、Hive部署&#xff08;手动部署&#xff09;下载Hive1. 上传安装包2. 解压Hive安装包2.1 解压2.2 重命名2.3 解决guava冲突 3. 配置Hive3.1 配置Hive环境变量3.2 修改 hive-site.xml 配置文件3.3 配置MySQL驱动包3.3.1 下在MySQL驱动包3.3.2 上传MySQL驱动包3.…...

c语言的qsort函数理解与使用

介绍&#xff1a;qsort 函数是 C 标准库中用于排序的快速排序算法函数。它的用法非常灵活&#xff0c;可以对任意类型的元素进行排序&#xff0c;只要提供了比较函数即可。 qsort 函数原型及参数解释&#xff1a; void qsort ( void* base, //指向要排序的数组的首元素…...

Java 语言的起源发展与基本概念(JDK,JRE,JVM)

Java语言的起源 源起 Java语言最初是由Sun Microsystems公司&#xff08;该公司于2009年被Oracle公司收购&#xff09;开发的一种编程语言。其创造者是詹姆斯高斯林&#xff08;James Gosling&#xff09;&#xff0c;他是一位加拿大计算机科学家。其前身名为Oak&#xff08;橡…...

03_变量

变量 var num 10; 变量的重新赋值 var num10; num 20; 变量提升 JavaScript 引擎的工作方式是&#xff0c;先解析代码&#xff0c;获取所有被声明的变量&#xff0c;然后再一行一行地运行。这造成的结果&#xff0c;就是所有的变量的声明语句&#xff0c;都会被提升到代码的…...

[论文阅读-综述]Supervised Speech Separation Based on Deep Learning: An Overview

基于深度学习的监督语音分离&#xff1a;综述 出版&#xff1a;IEEE 核心&#xff1a;使用语音分离将目标语音信号与噪声混合分离的计算 本文用于对该文章的学习&#xff0c;主要是对内容的理解翻译与笔记 1. 语音分离介绍 语音分离的目标&#xff1a;将目标语音与背景干扰分…...

群控系统服务端开发模式-应用开发-邮箱配置功能开发

邮箱配置主要是将管理员数据做归属。具体见下图&#xff1a; 一、创建表 1、语句 CREATE TABLE cluster_control.nc_param_mail (id int(11) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 编号,title varchar(20) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL COMMENT…...

【机器学习】——卷积与循环的交响曲:神经网络模型在现代科技中的协奏

&#x1f3bc;个人主页&#xff1a;【Y小夜】 &#x1f60e;作者简介&#xff1a;一位双非学校的大二学生&#xff0c;编程爱好者&#xff0c; 专注于基础和实战分享&#xff0c;欢迎私信咨询&#xff01; &#x1f386;入门专栏&#xff1a;&#x1f387;【MySQL&#xff0…...

android studio引用so库

在工程中编译好的so库文件将在原始编译工程对应目录下&#xff1a;build/intermediates/cxx/Debug/xxxxxx/obj/ 其目录结构如上所示&#xff0c;包含生成的四个版本&#xff0c;每个文件夹下均包含c/c源码编译成的Android版本的libnavi.so库和提供应用接口的libnavi-lib.so库。…...

TrollInstallerX终极指南:3分钟搞定iOS 14-16.6.1 TrollStore安装

TrollInstallerX终极指南&#xff1a;3分钟搞定iOS 14-16.6.1 TrollStore安装 【免费下载链接】TrollInstallerX A TrollStore installer for iOS 14.0 - 16.6.1 项目地址: https://gitcode.com/gh_mirrors/tr/TrollInstallerX TrollInstallerX是当前iOS 14.0至16.6.1设…...

PIC18F4550微控制器实现USB大容量存储设备设计

1. USB大容量存储设备设计概述USB大容量存储设备&#xff08;Mass Storage Device&#xff0c;MSD&#xff09;已成为现代数字生活中不可或缺的组成部分。从U盘到移动硬盘&#xff0c;这类设备的核心都是基于USB Mass Storage Class协议实现的。本文将深入探讨如何利用PIC18F45…...

联发科2012年崛起:从功能机到智能机的转型与挑战

1. 从功能机到智能机的惊险一跃&#xff1a;联发科的2012年2012年&#xff0c;对于全球移动芯片行业来说&#xff0c;是几家欢喜几家愁的一年。诺基亚和黑莓的持续衰落&#xff0c;直接拖垮了像ST-Ericsson这样深度绑定的芯片供应商&#xff1b;即便是巨头如高通&#xff0c;也…...

ARM TPIU调试接口原理与应用实践

1. ARM TPIU调试接口深度解析在嵌入式系统开发中&#xff0c;调试接口的设计与实现往往是决定开发效率的关键因素。作为ARM CoreSight调试架构的重要组成部分&#xff0c;Trace Port Interface Unit(TPIU)承担着处理器跟踪数据格式化与输出的核心功能。本文将深入剖析TPIU的寄存…...

【2026年携程暑期实习- 5月10日-第四题-单数组交换】(题目+思路+JavaC++Python解析+在线测试)

题目内容 游游有两个长度同为 nnn 的整数数组 aaa 和 bbb。她会对数组...

FuSa DFMEA在芯片验证中的借鉴价值

功能安全&#xff08;Functional Safety, FuSa&#xff09;领域的DFMEA&#xff08;Design Failure Mode and Effects Analysis&#xff0c;设计失效模式与影响分析&#xff09;是一种以预防为主的系统化、结构化风险管理方法&#xff0c;它通过分析失效模式并优化来降低风险。…...

Void编辑器:轻量级插件化架构与LSP/Tree-sitter深度集成解析

1. 项目概述&#xff1a;一个为“创造者”而生的现代编辑器最近在开发者社区里&#xff0c;一个名为“Void”的编辑器项目引起了我的注意。它不像那些我们耳熟能详的庞然大物&#xff0c;比如 VS Code 或 Sublime Text&#xff0c;一上来就带着庞大的生态和复杂的功能。Void 给…...

告别环境配置噩梦:用Shell脚本一键搞定VCS与Verdi的联调环境

芯片验证工程师的效率革命&#xff1a;Shell脚本全自动构建VCSVerdi联调环境 每次开始新项目都要重复配置验证环境&#xff1f;还在为VCS编译选项和Verdi波形调试的手动操作浪费时间&#xff1f;资深验证工程师的日常&#xff0c;不该被这些重复劳动占据。本文将带你用Shell脚本…...

ANSYS Workbench网格划分进阶:扫掠、多区与2D网格的实战精解

1. 扫掠网格划分&#xff1a;从原理到实战技巧 第一次用ANSYS Workbench做薄壁结构分析时&#xff0c;我对着那个复杂的几何模型发呆了半小时——到底该选哪种网格划分方法&#xff1f;直到掌握了扫掠网格的精髓&#xff0c;才发现原来处理这类问题可以如此高效。扫掠网格特别适…...

双模型协同工作流架构解析:从感知到决策的AI工程实践

1. 项目概述&#xff1a;双模型协同工作流的深度解构最近在GitHub上看到一个挺有意思的项目&#xff0c;叫“openclaw-dual-model-workflow”。光看这个名字&#xff0c;就能嗅到一股浓浓的工程实践和架构设计的味道。这不像是一个简单的应用Demo&#xff0c;更像是一个为解决特…...