【数据结构】—— 树和二叉树

  • 1、树的概念
  • 2、树的相关术语
  • 3、树的常见表示方法
  • 4、树的实际应用
  • 5、二叉树的相关概念和性质
  • 6、二叉树的顺序存储(堆)
    • 6.1 堆的概念
    • 6.2 堆的结构和接口
    • 6.3 堆的初始化和销毁
    • 6.4 堆的插入
    • 6.5 堆的删除
    • 6.5 取堆顶数据
    • 6.6 获取有效节点个数
    • 6.7 判空
    • 6.8 源代码
  • 7、堆的应用
    • 7.1 堆排序
    • 7.2 TopK
  • 8、二叉树的链式结构
    • 8.1 基本概念
    • 8.2 结构和接口
    • 8.3 二叉树的创建
    • 8.4 二叉树的遍历
      • 前序遍历
      • 中序遍历
      • 后序遍历
    • 8.5 二叉树节点个数
    • 8.6 二叉树的深度/高度
    • 8.7 二叉树叶节点个数
    • 8.8 二叉树第k层节点个数
    • 8.9 二叉树查找值为x的节点
    • 8.10 层序遍历
    • 8.11 判断是否为完全二叉树
    • 8.12 销毁
    • 8.13 源代码
  • 总结和思考

1、树的概念

在这里插入图片描述

树是一种非线性的数据结构,它是由n(n>=0) 个有限结点组成一个具有层次关系的集合,当n=0时被称为空树。
树的特点

  • 只有一个特殊的结点,称为根结点,根节点没有前驱结点
  • 树形结构中,除根结点以外,其余结点都被分成M(M>0)个互不相交的集合T1、T2、……、Tm,其中每一个集合Ti(1<=i<=m)又是一棵结构与树类似的子树。每棵子树的根结点(所有节点)有且只有一个前驱,可以有0个或多个后继结点。则n个结点的树有n-1条边
  • 树是递归定义的

2、树的相关术语

在这里插入图片描述

  • 节点的度:一个节点含有的子树的个数称为该节点的度
  • 叶节点或终端节点:度为0的节点称为叶节点,如上图:B、C、H、I、P…等节点
  • 非终端节点或分支节点:度不为0的节点,如上图:D、E、F、G…等节点
  • 双亲节点或父节点:若一个节点含有子节点,则这个节点被称为其子节点的父节点,如F是K和L的父节点
  • 孩子节点或子节点:一个节点含有子树的根节点称为该节点的子节点,如J是E的子节点
  • 兄弟节点:具有相同父节点的节点互称为兄弟节点,如D和E,I和J等等
  • 树的度:一棵树中,最大的节点的度称为树的度,如上图度为6
  • 节点的层次:从根开始定义,根为第1层,根的子节点为第2层,以此类推
  • 树的深度或高度:树中节点的最大层次,如上图:树的高度为4
  • 堂兄弟节点:双亲在同一层的互为堂兄弟;如上图:H和I,他们的父亲在同一层
  • 节点的祖先:从根到该节点所经分支上所有节点的祖先。如上图:A是图中所有节点的祖先
  • 子孙:以某节点为根的所有子树的节点称为该节点的子孙。如上图:所有节点都是A的子孙
  • 森林:由m棵互不相交的树的集合称为森林

3、树的常见表示方法

(1)链式结构(左孩子右兄弟)

typedef struct TreeNode {int data;struct TreeNode* left;//左孩子struct TreeNode* right;//右兄弟
}TNode;//链式结构是最常见的表示结构

优点

  • 无需浪费空间:链表动态分配内存,这样在构建树时不需要预先知道树的大小,可以有效利用内存。

  • 插入删除效率高 :在链式结构中,插入或删除节点不需要移动大量数据,只需调整指针,操作时间复杂度为 O(1),使得这些操作相对高效。因此在树的结构变化频繁的场景下,链式结构表示更为合适

缺点

  • 不适合需要频繁执行全树遍历或计算树的深度、宽度等操作的场景。
  • 节点除了存储数据外,还需要存储指向子节点的指针,增加了额外的内存开销

在这里插入图片描述

(2)动态顺序结构

typedef struct Heap
{HPDataType* a;//相当于数组存储int size;int capacity;//开辟的空间,用于判断是否需要扩容
}HP;

优点

  • 简单高效的访问:可以通过索引快速访问任何节点,通常时间复杂度为 O(1)
  • 节省空间:无需额外申请节点

缺点:插入、删除操作及频繁调整树结构时需要移动大量元素,特别是数组的头部和中间位置,时间复杂度为O(n)

(3)静态顺序结构

#define MAX_SIZE 100int tree[MAX_SIZE];  // 假设树的节点存储在数组中

适合完全二叉树满二叉树

(4)邻接表表示法

typedef struct Node {int data;                   // 节点数据struct Node** children;     // 指向子节点的指针数组int childCount;             // 子节点数量
} Nd;

邻接表表示法在处理树结构时非常高效,尤其是在动态结构空间利用方面表现突出
(5)链表数组表示法

#define MAX_LEVELS 10typedef struct TreeNode {int data;struct TreeNode* next;
} TreeNode;TreeNode* levelArray[MAX_LEVELS];

链表数组表示法结合了数组和链表的优点,适用于需要高效存储树结构的场景,特别是当树的结构较为稀疏时(和普通链式结构二叉树原理类似)

4、树的实际应用

  • 文件系统
    操作系统中的文件系统通常使用树结构来组织文件和目录。根目录是树的根节点,子目录和文件作为树的子节点,形成层级关系。
    在这里插入图片描述

  • 数据库索引
    数据库管理系统(DBMS)使用树结构(如B树或B+树)来索引数据表。索引结构帮助快速定位表中的记录。
    在这里插入图片描述

5、二叉树的相关概念和性质

在这里插入图片描述

二叉树是一种特殊的树形结构
二叉树需要满足两个条件:

  • 树形结构
  • 所有节点的度<=2

对于任意的二叉树都是由以下五种情况复合而成的

在这里插入图片描述
满二叉树:每一层节点均为最大节点数的二叉树。
完全二叉树:前h-1层为满二叉树,第h层自左向右是连续的二叉树。

在这里插入图片描述
二叉树的节点数和深度计算

  • 二叉树第i层的结点个数:若规定根结点的层数为 1 ,则一棵非空二叉树的第i层上最多有 2i-1 个结点
  • 二叉树的总结点个数:若规定根结点的层数为 1 ,则深度为 h 的二叉树的最大结点数是 2h-1
  • 满二叉树的深度:若规定根结点的层数为 1 ,则具有 n 个结点的满二叉树的深度为h = log2 (n + 1)
  • 二叉树的叶节点个数和度为2节点个数的关系h0=h2+1

叶节点个数代表二叉树的分支个数
在这里插入图片描述

6、二叉树的顺序存储(堆)

(实现小根堆)

6.1 堆的概念

使用数组存储二叉树即为堆

堆可以分为两种主要类型
最大堆:每个节点的关键码都大于或等于其子节点的关键码,即根节点具有最大的关键码
最小堆:每个节点的关键码都小于于或等于其子节点的关键码,即根节点具有最小的关键码

最大堆(大根堆)
在这里插入图片描述

最小堆(小根堆)
在这里插入图片描述

6.2 堆的结构和接口

typedef int HPDataType;typedef struct Heap
{HPDataType* a;int size;int capacity;
}HP;//初始化
void HeapInit(HP* php);
//销毁
void HeapDestroy(HP* php);
//插入
void HeapPush(HP* php, HPDataType x);
// 规定删除堆顶(根节点)
void HeapPop(HP* php);
//获取堆顶元素
HPDataType HeapTop(HP* php);
//获取堆有效元素个数
size_t HeapSize(HP* php);
//判空
bool HeapEmpty(HP* php);
//用于交换节点
void Swap(HPDataType* p1, HPDataType* p2);
//向上调整
void AdjustUp(HPDataType* a, int child);
//向下调整
void AdjustDown(HPDataType* a, int size, int parent);

6.3 堆的初始化和销毁

和顺序表类似

void HeapInit(HP* php)
{assert(php);php->a = NULL;php->size = 0;php->capacity = 0;
}void HeapDestroy(HP* php)
{assert(php);free(php->a);php->a = NULL;php->size = php->capacity = 0;
}

6.4 堆的插入

用向上调整算法遍历整个数组,交换节点位置,使新节点插入到合适的位置

void Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}void AdjustUp(HPDataType* a, int child)
{int parent = (child - 1) / 2;while (child > 0){if (a[child] < a[parent]){Swap(&a[child], &a[parent]);child = parent;parent = (child - 1) / 2;}else{break;}}
}// O(logN)
void HeapPush(HP* php, HPDataType x)
{assert(php);if (php->size == php->capacity){int newCapacity = php->capacity == 0 ? 4 : php->capacity * 2;HPDataType* tmp = (HPDataType*)realloc(php->a, newCapacity * sizeof(HPDataType));if (tmp == NULL){perror("realloc fail");exit(-1);}php->a = tmp;php->capacity = newCapacity;}php->a[php->size] = x;php->size++;AdjustUp(php->a, php->size - 1);
}

向上调整算法分析:
如何获取父节点:parent=(child-1/)2;
将新节点插入到堆底,随后和父节点比较大小,如果小于父节点则交换两个节点,更新父节点和孩子节点位置,再继续向上调整,直到堆中所有父节点都小于孩子节点。否则直接退出循环
循环条件:while(chilid>0);一直调整到根节点
在这里插入图片描述

6.5 堆的删除

堆的删除时删除堆顶数据。将堆底元素覆盖堆顶,对堆顶元素使用向下调整算法即可

void AdjustDown(HPDataType* a, int size, int parent)
{int child = parent * 2 + 1;while (child < size){// 假设左孩子小,如果解设错了,更新一下if (child + 1 < size && a[child + 1] < a[child]){++child;}if (a[child] < a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}void HeapPop(HP* php)
{assert(php);assert(php->size > 0);Swap(&php->a[0], &php->a[php->size - 1]);php->size--;AdjustDown(php->a, php->size, 0);
}

向下调整算法分析
已知一个父节点parent算出左右孩子节点:

  • 左孩子:child = 2 * parent + 1
  • 右孩子:child = 2 * parent + 2
    在这里插入图片描述

向下调整算法:如果建的是小堆,首先从根节点作为父结点开始与其左右孩子进行比较,如果父节点大于左孩子节点或者大于右孩子节点(如果都大于则与最小的孩子节点进行交换),然后更新父节点和孩子节点,继续向下调整,如果父节点小于左右孩子节点则停止,使得所有的父节点都小于或等于孩子节点。如果建的是大堆,则反之。

注意:当向下调整到最后一层的时候就停止,因为最后一层如果作为父节点是没有孩子节点的,所以不用继续向下调整了,所以循环条件为child<n

6.5 取堆顶数据

返回堆顶即可

HPDataType HeapTop(HP* php)
{assert(php);assert(php->size > 0);return php->a[0];
}

6.6 获取有效节点个数

返回size即可

size_t HeapSize(HP* php)
{assert(php);return php->size;
}

6.7 判空

判断size的大小即可

bool HeapEmpty(HP* php)
{assert(php);return php->size == 0;
}

6.8 源代码

.h头文件

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
#include<time.h>typedef int HPDataType;typedef struct Heap
{HPDataType* a;int size;int capacity;
}HP;void HeapInit(HP* php);
void HeapDestroy(HP* php);
void HeapPush(HP* php, HPDataType x);
// 规定删除堆顶(根节点)
void HeapPop(HP* php);
HPDataType HeapTop(HP* php);
size_t HeapSize(HP* php);
bool HeapEmpty(HP* php);void Swap(HPDataType* p1, HPDataType* p2);
void AdjustUp(HPDataType* a, int child);
void AdjustDown(HPDataType* a, int size, int parent);void TopK1(int* a, int n, int k);
void TopK2(int* a, int n, int k);

.c源文件

#include"Heap.h"// 小堆
void HeapInit(HP* php)
{assert(php);php->a = NULL;php->size = 0;php->capacity = 0;
}void HeapDestroy(HP* php)
{assert(php);free(php->a);php->a = NULL;php->size = php->capacity = 0;
}void Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}//20:19jixu 
void AdjustUp(HPDataType* a, int child)
{int parent = (child - 1) / 2;//while (parent >= 0)while (child > 0){if (a[child] < a[parent]){Swap(&a[child], &a[parent]);child = parent;parent = (child - 1) / 2;//child = (child - 1) / 2;//parent = (parent - 1) / 2;}else{break;}}
}// O(logN)
void HeapPush(HP* php, HPDataType x)
{assert(php);if (php->size == php->capacity){int newCapacity = php->capacity == 0 ? 4 : php->capacity * 2;HPDataType* tmp = (HPDataType*)realloc(php->a, newCapacity * sizeof(HPDataType));if (tmp == NULL){perror("realloc fail");exit(-1);}php->a = tmp;php->capacity = newCapacity;}php->a[php->size] = x;php->size++;AdjustUp(php->a, php->size - 1);
}// 21:15继续void AdjustDown(HPDataType* a, int size, int parent)
{int child = parent * 2 + 1;while (child < size){// 假设左孩子小,如果解设错了,更新一下if (child + 1 < size && a[child + 1] > a[child]){++child;}//大堆if (a[child] > a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}void HeapPop(HP* php)
{assert(php);assert(php->size > 0);Swap(&php->a[0], &php->a[php->size - 1]);php->size--;AdjustDown(php->a, php->size, 0);
}HPDataType HeapTop(HP* php)
{assert(php);assert(php->size > 0);return php->a[0];
}size_t HeapSize(HP* php)
{assert(php);return php->size;
}bool HeapEmpty(HP* php)
{assert(php);return php->size == 0;
}void TopK1(int* a, int n, int k)
{for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, k, i);}printf("排序后:");//堆排序int end = n - 1;while (k > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);printf("%d ", a[end]);--end;k--;}
}void TopK2(int* a, int n, int k)
{for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, k, i);}printf("排序后:");//topkfor (int i = k; i < n; i++){if (a[i] < a[0]){Swap(&a[i], &a[0]);AdjustDown(a, k, 0);}}
}

测试文件

#include"Heap.h"//int main()
//{
//	int a[] = { 4,6,2,1,5,8,2,9};
//	HP hp;
//	HeapInit(&hp);
//	for (int i = 0; i < sizeof(a) / sizeof(int); ++i)
//	{
//		HeapPush(&hp, a[i]);
//	}
//
//	/*int k = 3;
//	while (k--)
//	{
//		printf("%d\n", HeapTop(&hp));
//		HeapPop(&hp);
//	}*/
//
//	while (!HeapEmpty(&hp))
//	{
//		printf("%d ", HeapTop(&hp));
//		HeapPop(&hp);
//	}
//	printf("\n");
//
//	return 0;
//}// 升序
//void HeapSort(int* a, int n)
//{
//	// 建大堆
//
//	// O(N*logN)
//	/*for (int i = 1; i < n; i++)
//	{
//		AdjustUp(a, i);
//	}*/
//
//	// 建堆 O(N)
//	//保证所有父节点大于等于子节点
//	for (int i = (n - 1 - 1) / 2; i >= 0; --i)
//	{
//		AdjustDown(a, n, i);
//	}
//	//堆排序
//	int end = n - 1;
//	while (end > 0)
//	{
//		Swap(&a[0], &a[end]);
//		AdjustDown(a, end, 0);
//		--end;
//	}
//}
//
//int main()
//{
//	int a[] = { 4, 6, 2, 1, 5, 8, 2, 9 };
//	printf("排序前:");
//	for (int i = 0; i < sizeof(a) / sizeof(int); i++)
//	{
//		printf("%d ", a[i]);
//	}
//	printf("\n");
//	HeapSort(a, sizeof(a)/sizeof(int));
//	printf("排序后:");
//	for (int i = 0; i < sizeof(a)/sizeof(int); i++)
//	{
//		printf("%d ", a[i]);
//	}
//	printf("\n");
//
//	return 0;
//}void HeapSort(int* a, int n)
{// 建大堆// O(N*logN)/*for (int i = 1; i < n; i++){AdjustUp(a, i);}*/// 建堆 O(N)//保证所有父节点大于等于子节点for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, n, i);}//堆排序int end = n-1;while (end > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);--end;}
}//int main()
//{
//	int a[] = { 4, 6, 2, 1, 5, 8, 2, 9 };
//	printf("排序前:");
//	for (int i = 0; i < sizeof(a) / sizeof(int); i++)
//	{
//		printf("%d ", a[i]);
//	}
//	printf("\n");
//	HeapSort(a, sizeof(a) / sizeof(int));
//	printf("排序后:");
//	for (int i = 0; i < sizeof(a) / sizeof(int); i++)
//	{
//		printf("%d ", a[i]);
//	}
//	printf("\n");
//
//	return 0;
//}int main()
{int a[] = { 4, 6, 10, 1, 5, 8, 2, 9 };printf("排序前:");for (int i = 0; i < sizeof(a) / sizeof(int); i++){printf("%d ", a[i]);}printf("\n");TopK2(a, sizeof(a) / sizeof(int), 3);for (int i = 0; i < 3; i++){printf("%d ", a[i]);}printf("\n");return 0;
}

7、堆的应用

1)堆排序
2)TopK

7.1 堆排序

在原数组排序

(1)将原数组调整为堆
(2)交换堆顶元素和堆尾元素
(3)堆顶进行向下调整操作
(4)调整元素个数end--;,使调整过的元素在正确位置,不再改变(此时堆底的i个元素已经在正确位置,因此不当作在堆里面)
(5)重复2~4步骤直到end=1,排序结束

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
#include<time.h>typedef int HPDataType;typedef struct Heap
{HPDataType* a;int size;int capacity;
}HP;void Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}void AdjustDown(HPDataType* a, int size, int parent)
{int child = parent * 2 + 1;while (child < size){// 假设左孩子小,如果解设错了,更新一下if (child + 1 < size && a[child + 1] < a[child]){++child;}if (a[child] < a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}
//升序建大堆
void HeapSort(int* a, int n)
{// 建大堆// O(N*logN)/*for (int i = 1; i < n; i++){AdjustUp(a, i);}*/// 建堆 O(N)//保证所有父节点大于等于子节点for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, n, i);}//堆排序int end = n - 1;while (end > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);--end;}
}int main()
{int a[] = { 4, 6, 2, 1, 5, 8, 2, 9 };printf("排序前:");for (int i = 0; i < sizeof(a) / sizeof(int); i++){printf("%d ", a[i]);}printf("\n");HeapSort(a, sizeof(a)/sizeof(int));printf("排序后:");for (int i = 0; i < sizeof(a)/sizeof(int); i++){printf("%d ", a[i]);}printf("\n");return 0;
}

7.2 TopK

最小/大的k个节点

当数据量过大的时候,数据只能存储在内存中,而内存中不能建堆,只能使用第二种方法

方法一对原数组进行操作

  • 时间复杂度:O(nlog2n)
  • 原理和堆排序相同:堆排序操作n-1次完成排序,TopK操作k次找出最大/小的k个节点
    找最大的k个节点用大堆

1)使用向下调整算法建堆
在这里插入图片描述

// 建堆 O(N)
//保证所有父节点大于等于子节点
for (int i = (n - 1 - 1) / 2; i >= 0; --i)
{AdjustDown(a, n, i);
}

2)取堆顶元素,与堆尾元素交换

Swap(&a[0],&a[end-1]);

3)从堆顶向下调整,选出堆中的最值,堆元素个数减1

AdjustDown(a, end, 0);
end--;

4)重复 2 3 步骤k次

以选出最大的3个节点分析

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

即选出最大的三个节点 99 85 66

完整代码

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>
#include<time.h>typedef int HPDataType;typedef struct Heap
{HPDataType* a;int size;int capacity;
}HP;void Swap(HPDataType* p1, HPDataType* p2)
{HPDataType tmp = *p1;*p1 = *p2;*p2 = tmp;
}void AdjustDown(HPDataType* a, int size, int parent)
{int child = parent * 2 + 1;while (child < size){// 假设左孩子小,如果解设错了,更新一下if (child + 1 < size && a[child + 1] > a[child]){++child;}if (a[child] > a[parent]){Swap(&a[child], &a[parent]);parent = child;child = parent * 2 + 1;}else{break;}}
}void TopK(int* a, int n, int k)
{for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, n, i);}printf("排序后:");//堆排序int end = n - 1;while (k > 0){Swap(&a[0], &a[end]);AdjustDown(a, end, 0);printf("%d ", a[end]);--end;k--;}
}

方法二:用前k个数据建小堆,后续数据和堆顶数据比较,如果比堆顶数据大,就替代堆顶,向下调整。遍历完数组结束(取小建大堆,取大建小)
具体步骤

((取最大k个数)建小堆 /(取最小k个数)建大堆 )

1) 取前k个数建堆
在这里插入图片描述
2)

  • 将剩余的n-k个节点和堆顶比较大小
  • 如果小于堆顶则入堆(替换堆顶),从堆顶进行一次向下调整(Adjustdown),否则不入堆
    在这里插入图片描述

得出最大/小的k个数(topk)
在这里插入图片描述

void TopK2(int* a, int n, int k)
{for (int i = (n - 1 - 1) / 2; i >= 0; --i){AdjustDown(a, k, i);}printf("排序后:");//topkfor (int i = k; i < n; i++){if (a[i] < a[0]){Swap(&a[i], &a[0]);AdjustDown(a, k, 0);}}
}

需要注意的取得的前k个小的数不一定是有序的
在这里插入图片描述
时间复杂度: O(n) = k + (n − k)log2 k

8、二叉树的链式结构

8.1 基本概念

用链表来表示一棵二叉树, 通常的方法是链表中每个结点由三个域组成:数据域和左右指针域,左右指针分别用来给出该结点左孩子和右孩子所在的链结点的存储地址。
在这里插入图片描述

8.2 结构和接口

typedef int BTDataType;typedef struct BinTreeNode
{struct BinTreeNode* left;struct BinTreeNode* right;int val;
}BTNode;
//创建新节点
BTNode* BuyBTNode(BTDataType x);
//前序遍历
void PreOrder(BTNode* root);
//中序遍历
void InOrder(BTNode* root);
//后序遍历
void PostOrder(BTNode* root);
//树的大小
int TreeSize(BTNode* root);
//树的深(高)度
int TreeHeight(BTNode* root);
//二叉树叶子结点个数
int TreeLeafSize(BTNode* root);
//树第k层的节点
int TreeKLevel(BTNode* root, int k);
//查找值为x的节点
BTNode* TreeFind(BTNode* root, int x);
//层序遍历
void TreeLevelOrder(BTNode* root);
//判断是否是完全二叉树
bool TreeisComplete(BTNode* root);
//销毁
void TreeDestroy(BTNode* root);

8.3 二叉树的创建

//创建二叉树新节点
BTNode* BuyBTNode(BTDataType x)
{BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));if (newnode == NULL){perror("malloc fail!");exit(1);}newnode->val = x;newnode->left = newnode->right = NULL;return newnode;
}BTNode* CreateTree()
{BTNode* n1 = BuyBTNode(1);BTNode* n2 = BuyBTNode(2);BTNode* n3 = BuyBTNode(3);BTNode* n4 = BuyBTNode(4);BTNode* n5 = BuyBTNode(5);BTNode* n6 = BuyBTNode(6);n1->left = n2;n1->right = n4;n2->left = n3;n4->left = n5;n4->right = n6;return n1;
}

创建二叉树如下
在这里插入图片描述

8.4 二叉树的遍历

前序遍历

//前序遍历: 根 左子树 右子树
void PreOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}printf("%d ", root->val);PreOrder(root->left);PreOrder(root->right);
}

前序遍历的结果:1 2 3 N N N 4 5 N N 6 N N

中序遍历

//中序遍历:左子树 根 右子树
void InOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}InOrder(root->left);printf("%d ", root->val);InOrder(root->right);
}

中序遍历的结果:N 3 N 2 N 1 N 5 N 4 N 6 N

后序遍历

//后序遍历:左子树 右子树  根
void PostOrder(BTNode* root)
{if (root == NULL){printf("N ");return;}PostOrder(root->left);PostOrder(root->right);printf("%d ", root->val);
}

后序遍历的结果:N N 3 N 2 N N 5 N N 6 4 1

8.5 二叉树节点个数

如果该root为NULL,则root不是节点,返回0,否则返回左子树和右子树的节点的总数+1即可

int TreeSize(BTNode* root)
{//静态变量不存在栈帧,存在于静态区//局部静态成员变量只会被初始化一次//static int size = 0;/*if (root == NULL) return 0;else ++size;TreeSize(root->left);TreeSize(root->right);return size;*///以上方法有线程安全的风险return root == NULL? 0 : TreeSize(root->left) + TreeSize(root->right)+1;
}

8.6 二叉树的深度/高度

如果root是NULL,则root不是节点,返回0,否则返回左子树和右子树的最大深度

int TreeHeight(BTNode* root)
{if (root == NULL){return 0;}int leftDepth = TreeHeight(root->left);int rightDepth = TreeHeight(root->right);return leftDepth > rightDepth ? leftDepth + 1 : rightDepth + 1;
}

8.7 二叉树叶节点个数

int TreeLeafSize(BTNode* root)
{//0个节点if (root == NULL){return 0;}//一个节点if (root->left == NULL && root->right == NULL){return 1;}//返回左子树叶节点和右子树叶节点个数总和return TreeLeafSize(root->left) + TreeLeafSize(root->right);
}

8.8 二叉树第k层节点个数

根节点起始位置在第一层,递归k-1次即可到达第k层,此时统计左子树和右子树的总数即可

int TreeKLevel(BTNode* root, int k)
{assert(k > 0);if (root == NULL) return 0;if (k == 1)return 1;//k>1 分治 转化为子问题求解return TreeKLevel(root->left, k - 1) + TreeKLevel(root->right, k - 1);
}

8.9 二叉树查找值为x的节点

BTNode* TreeFind(BTNode* root, int x)
{if (root == NULL)return NULL;if (root->val == x)return root;没找到——>递归子问题这样写会造成返回值丢失(能找到指定节点,但是无法返回该节点)//TreeFind(root->left, x);//TreeFind(root->right, x);//为了解决返回值丢失和减少递归次数//应当创建节点接收返回值,减少递归次数,降低时间复杂度BTNode* ret1 = TreeFind(root->left, x);//如果左节点找到了(有返回值)直接返回即可if (ret1)return ret1;BTNode* ret2 = TreeFind(root->right, x);//如果右节点找到了(有返回值)直接返回即可if (ret2)return ret2;return NULL;//整个树都没找到,返回NULL
}

8.10 层序遍历

利用队列实现一层一层遍历(FIFO)
本质上是广度优先遍历(bfs)

void TreeLevelOrder(BTNode* root)
{Queue q;QueueInit(&q);if (root != NULL){QueuePush(&q, root);}//从上至下while (!QueueEmpty(&q)){BTNode* top = QueueFront(&q);QueuePop(&q);if (top){printf("%d ", top->val);//从左向右if (top->left){QueuePush(&q, top->left);}if (top->right){QueuePush(&q, top->right);}}else printf("N ");}printf("\n");QueueDestroy(&q);
}

在这里插入图片描述

8.11 判断是否为完全二叉树

利用层序遍历完成

//用层序遍历实现
bool TreeisComplete(BTNode* root)
{Queue q;QueueInit(&q);if (root != NULL){QueuePush(&q, root);}//第一个循环结束条件为队列为空或者出现空节点while (!QueueEmpty(&q)){BTNode* front = QueueFront(&q);QueuePop(&q);if (front == NULL) break;//无论左子树右子树是否为空都要加入队列QueuePush(&q, front->left);QueuePush(&q, front->right);}//队列非空说明出现了空节点//如果存在一个节点不是空节点——>说明不是完全二叉树//否则则是完全二叉树while (!QueueEmpty(&q)){BTNode* front = QueueFront(&q);QueuePop(&q);if (front != NULL){QueueDestroy(&q);return false;}}QueueDestroy(&q);return true;
}

8.12 销毁

从根节点开始,递归左子树和右子树销毁所有节点

void TreeDestroy(BTNode* root)
{if (root == NULL)return;TreeDestroy(root->left);TreeDestroy(root->right);free(root);
}

8.13 源代码

.h头文件

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>4
#include<stdbool.h>
typedef int BTDataType;typedef struct BinTreeNode
{struct BinTreeNode* left;struct BinTreeNode* right;BTDataType val;
}BTNode;
BTNode* BuyBTNode(BTDataType x);
//前序遍历
void PreOrder(BTNode* root);
//中序遍历
void InOrder(BTNode* root);
//后序遍历
void PostOrder(BTNode* root);
//树的大小
int TreeSize(BTNode* root);
//树的深度
int TreeHeight(BTNode* root);
//二叉树叶子结点个数
int TreeLeafSize(BTNode* root);
//树第k层的节点
int TreeKLevel(BTNode* root, int k);
//查找值为x的节点
BTNode* TreeFind(BTNode* root, int x);
//层序遍历
void TreeLevelOrder(BTNode* root);
//判断是否为完全二叉树
bool TreeisComplete(BTNode* root);
//销毁
void TreeDestroy(BTNode* root);

.c源文件

#include "Bintree.h"
#include "Queue.h"
static int size = 0;
//创建二叉树新节点
BTNode* BuyBTNode(BTDataType x)
{BTNode* newnode = (BTNode*)malloc(sizeof(BTNode));if (newnode == NULL){perror("malloc fail!");exit(1);}newnode->val = x;newnode->left = newnode->right = NULL;return newnode;
}
//前序遍历: 根 左子树 右子树
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);
}int TreeSize(BTNode* root)
{//静态变量不存在栈帧,存在于静态区//局部静态成员变量只会被初始化一次//static int size = 0;/*if (root == NULL) return 0;else ++size;TreeSize(root->left);TreeSize(root->right);return size;*///以上方法有线程安全的风险return root == NULL? 0 : TreeSize(root->left) + TreeSize(root->right)+1;
}int TreeHeight(BTNode* root)
{if (root == NULL){return 0;}int leftDepth = TreeHeight(root->left);int rightDepth = TreeHeight(root->right);return leftDepth > rightDepth ? leftDepth + 1 : rightDepth + 1;
}int TreeLeafSize(BTNode* root)
{if (root == NULL){return 0;}if (root->left == NULL && root->right == NULL){return 1;}//返回左子树叶节点和右子树叶节点个数总和return TreeLeafSize(root->left) + TreeLeafSize(root->right);
}int TreeKLevel(BTNode* root, int k)
{assert(k > 0);if (root == NULL) return 0;if (k == 1)return 1;//k>1 分治 转化为子问题求解return TreeKLevel(root->left, k - 1) + TreeKLevel(root->right, k - 1);
}BTNode* TreeFind(BTNode* root, int x)
{if (root == NULL)return NULL;if (root->val == x)return root;没找到——>递归子问题这样写会造成返回值丢失(能找到指定节点,但是无法返回该节点)//TreeFind(root->left, x);//TreeFind(root->right, x);//为了解决返回值丢失和减少递归次数//应当创建节点接收返回值,cong'er 减少递归次数,降低时间复杂度BTNode* ret1 = TreeFind(root->left, x);//如果左节点找到了(有返回值)直接返回即可if (ret1)return ret1;BTNode* ret2 = TreeFind(root->right, x);//如果右节点找到了(有返回值)直接返回即可if (ret2)return ret2;return NULL;//整个树都没找到,返回NULL
}
//层序遍历一般采用队列实现
//本质上是广度优先遍历(bfs)
void TreeLevelOrder(BTNode* root)
{Queue q;QueueInit(&q);if (root != NULL){QueuePush(&q, root);}//从上至下while (!QueueEmpty(&q)){BTNode* top = QueueFront(&q);QueuePop(&q);if (top){printf("%d ", top->val);//从左向右if (top->left){QueuePush(&q, top->left);}if (top->right){QueuePush(&q, top->right);}}else printf("N ");}printf("\n");QueueDestroy(&q);
}
//用层序遍历实现
bool TreeisComplete(BTNode* root)
{Queue q;QueueInit(&q);if (root != NULL){QueuePush(&q, root);}//第一个循环结束条件为队列为空或者出现空节点while (!QueueEmpty(&q)){BTNode* front = QueueFront(&q);QueuePop(&q);if (front == NULL) break;//无论左子树右子树是否为空都要加入队列QueuePush(&q, front->left);QueuePush(&q, front->right);}//队列非空说明出现了空节点//如果接下来出现一个节点不是空节点---->说明不是完全二叉树,否则则是完全二叉树while (!QueueEmpty(&q)){BTNode* front = QueueFront(&q);QueuePop(&q);if (front != NULL){QueueDestroy(&q);return false;}}QueueDestroy(&q);return true;
}void TreeDestroy(BTNode* root)
{if (root == NULL)return;TreeDestroy(root->left);TreeDestroy(root->right);free(root);
}

测试文件

#include"Bintree.h"BTNode* CreateTree()
{BTNode* n1 = BuyBTNode(1);BTNode* n2 = BuyBTNode(2);BTNode* n3 = BuyBTNode(3);BTNode* n4 = BuyBTNode(4);BTNode* n5 = BuyBTNode(5);BTNode* n6 = BuyBTNode(6);n1->left = n2;n1->right = n4;n2->left = n3;n4->left = n5;n4->right = n6;return n1;
}int main()
{BTNode* root = CreateTree();PreOrder(root);printf("\n");InOrder(root);printf("\n");PostOrder(root);printf("\n");printf("二叉树的节点个数:%d \n", TreeSize(root));printf("二叉树的第3层节点个数:%d \n", TreeKLevel(root,3));printf("二叉树的大小是:%d \n", TreeSize(root));printf("二叉树的高度是:%d\n", TreeHeight(root));if (TreeFind(root, 5))printf("找到了\n");elseprintf("没找到\n");printf("二叉树的叶子节点个数:%d \n",TreeLeafSize(root));TreeLevelOrder(root);if (TreeisComplete(root)) printf("完全二叉树");else printf("不完全二叉树");printf("\n");TreeDestroy(root);root = NULL;//外部置空return 0;
}

总结和思考

  • 二叉树是非线性数据结构,相较于栈队列链表等线性结构更加抽象。为了更好的理解,可以尝试手动画递归展开图,将接口的每个栈帧都具象化。
  • 至于二叉树笔试题目,我们可以先手动推导一遍,再去核对公式,有利于更深刻理解公式的原理。
    END

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

高并发业务下的库存扣减技术方案设计

扣减库存需要查询库存是否足够: 足够就占用库存不够则返回库存不足&#xff08;这里不区分库存可用、占用、已消耗等状态&#xff0c;统一成扣减库存数量&#xff0c;简化场景&#xff09; 并发场景&#xff0c;若 查询库存和扣减库存不具备原子性&#xff0c;就可能超卖&…

动态内存管理函数malloc,calloc,realloc,free

malloc 函数原型&#xff1a;void* malloc(size_t size); 这个函数向内存申请一块连续可用的size大小的空间&#xff0c;并返回指向这快空间的指针。如果开辟成功&#xff0c;则返回一个指向开辟好空间的指针。如果开辟失败&#xff0c;则返回一个NULL指针&#xff0c;因此ma…

Facebook AI策略全解:从数据分析到智能推荐的成功秘诀

在数字化时代&#xff0c;人工智能&#xff08;AI&#xff09;已成为推动科技发展的核心力量。Facebook&#xff0c;作为全球领先的社交网络平台&#xff0c;正通过先进的AI策略来优化用户体验和平台运营。从数据分析到智能推荐&#xff0c;Facebook的AI策略涵盖了多个方面&…

Git 分支操作全解析:创建、切换、合并、删除及冲突解决

“ 在现代软件开发中&#xff0c;高效的版本控制是确保项目成功的关键。Git 提供了强大的分支管理功能&#xff0c;使得开发者能够独立地进行功能开发、修复 bug 和进行紧急修补。本文将深入探讨 Git 分支的基本操作&#xff0c;包括创建、切换、合并和删除分支&#xff0c;同时…

Linux基础 - yum、rzsz、vim 使用与配置、gcc/g++的详细解说

目录 一、Linux 软件包管理器 yum A.什么是软件包&#xff1f; B.关于rzsz&#xff0c;yum的配置 1.安装 sz&#xff0c;rz 命令&#xff1a; a.执行命令sz可将linux中的文件传输到Windows中 b.执行rz命令可将Windows中的文件传输到linux 2.scp XXX.tgz 用户名另一台lin…

免费高画质提取PPT/Word/Excel中的图片工具

下载地址&#xff1a;https://pan.quark.cn/s/134ccc35b8a2 软件简介&#xff1a; 好不容易搞到一个几十上百MB的ppt&#xff0c;想导出里面的图片进行二次加工&#xff0c;却被ppt超低画质的图片另存为功能劝退&#xff0c;明知里面全是高清图片&#xff0c;走时却是两手空空…

1系-8系铝合金材料的成分特性及应用详解

1系-8系铝合金材料的成分特性及应用详解 铝合金概述 铝合金的定义铝合金是一种以铝为基体&#xff0c;通过添加一定量的其他合金化元素&#xff08;如铜、锰、硅、镁、锌等&#xff09;形成的合金材料。由于合金元素的加入&#xff0c;铝合金在保持铝的轻质、良好导电导热性等基…

langchain入门系列之六 使用langchain构建PDF解析助手

本文将介绍如何使用langchain构建一个pdf解析助手&#xff0c;在此文中你将学习到langchain如何与web应用(fastapi)相结合&#xff0c;向量持久化等知识&#xff0c;话不多说&#xff0c;现在开始。 安装环境 pip install fastapi pip install python-dotenv pip install uv…

漫步者这款耳机怎么样吗?南卡、漫步者、Cleer公认畅销款式测评!

目前市场上开放式耳机品牌众多&#xff0c;选择时需要充分了解&#xff0c;但即便如此&#xff0c;也难以完全避免购买到质量不佳的产品。作为一位专注于数码产品测评的博主&#xff0c;我对开放式耳机有深入的研究。最近&#xff0c;我收到了许多关于漫步者、南卡、Cleer等品牌…

Flutter-自适用高度PageView

需求 在 Flutter 中&#xff0c;PageView 是一个非常常用的组件&#xff0c;能够实现多个页面的滑动切换。然而&#xff0c;默认的 PageView 高度是固定的&#xff0c;这在展示不同高度的页面时&#xff0c;可能会导致不必要的空白或内容裁剪问题。为了使 PageView 能够根据每…

OpenMax算法详解:深度学习中的高效开集识别技术

OpenMax算法详解&#xff1a;深度学习中的高效开集识别技术 在深度学习领域&#xff0c;模型的识别能力往往受限于其训练数据集的范畴。传统的分类模型&#xff0c;如卷积神经网络&#xff08;CNN&#xff09;或循环神经网络&#xff08;RNN&#xff09;&#xff0c;通常被设计…

第八节:Nodify 编辑器属性

引言 经过前几章的学习&#xff0c;你已经对Nodify框架有了初步的编程思路。当然只局限于这些还完全不够&#xff0c;本章节将阐述各个结构组件的一些常用属性&#xff0c;以便在日后的开发过程中更得心应手。 1、编辑器 平移 简介属性默认值平移功能 控制DisablePanningfals…

100128-批量获取视频音频时长添加到文件名中支持子孙文件夹下操作-UI

程序功使用环境▶适用的系统环境说明&#xff1a;win7以上64位win系统注意&#xff1a;win32位系统/mac系统需要额外定制▶使用期限&#xff1a;无需注册、不绑电脑、无时间限制▶如何安装&#xff1a;不需要安装程序功能说明▶子文件夹穿透&#xff1a;支持▶支持的文件格式&a…

MySQL集群技术详解

目录 一、MySQL在服务器中的部署方法 1.1 编译安装MySQL 1.2 部署MySQL 二、MySQL主从复制 2.1 配置master 2.2 配置slave 2.3 添加slave2 测试&#xff1a; 2.4 延迟复制 2.5 慢查询日志 2.6 MySQL的并行复制 2.7 MySQL主从复制原理剖析 2.8 架构缺陷 三、MySQL…

学习笔记——IP组播——IP组播基本概述

二、IP组播基本概述 IP组播技术有效地解决了单播和广播在点到多点应用中的问题。组播源只发送一份数据&#xff0c;数据在网络节点间被复制、分发&#xff08;PIM&#xff09;&#xff0c;且只发送给需要该信息的接收者。 1、前言 网络中存在各种各样的业务&#xff0c;从流…

EasyCVR视频汇聚平台革新播放体验:WebRTC协议赋能H.265视频流畅传输

随着科技的飞速发展和网络技术的不断革新&#xff0c;视频监控已经广泛应用于社会各个领域&#xff0c;成为现代安全管理的重要组成部分。在视频监控领域&#xff0c;视频编码技术的选择尤为重要&#xff0c;它不仅关系到视频的质量&#xff0c;还直接影响到视频的传输效率和兼…

企业参与制定行业标准的主要途径有哪些?需要具备哪些条件?

在当今竞争激烈的商业环境中&#xff0c;参与制定行业标准已成为企业提升竞争力、塑造行业地位的重要战略举措。然而&#xff0c;并非所有企业都有能力和资格参与这一重要的活动。要想在行业标准制定的舞台上发挥积极作用&#xff0c;企业需要具备一系列关键条件。 企业参与制…

mapstruct和lombok同时使用时,转换实体类时数据丢失

全局搜一下maps&#xff0c;找到你进行转换的方法 可以看到新建了TswCaseInfoPlus后直接返回了&#xff0c;说明TswCaseInfoPlus没有set方法&#xff0c;或者说编译后lombok没生效 在pom文件中&#xff0c;编译打包插件中将lombok&#xff0c;mapstruct&#xff0c;lombok-map…

3ds Max - 导出顶点色模型

很久之前的笔记&#xff0c;整理归档&#xff1b; 在3ds Max中&#xff0c;给模型添加VetexPaint修改器后&#xff0c;可以给模型&#xff08;顶点色通道R\G\B默认值为255\255\255&#xff09;刷不同颜色的顶点色&#xff08;默认为黑色&#xff0c;即让RGB通道都为0&#xff0…

PY信号和槽

知不足而奋进 望远山而前行 提示&#xff1a;写完文章后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 系列文章目录前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 在使用PyQt进行图形用户界面&#xff08;GU…