二叉树的顺序存储

目录

顺序存储:

简介:

节点的位置关系:

优缺点:

优点:

缺点:

二叉树顺序存储的模拟实现:

向上调整算法:

向下调整算法:

二叉树的初始化:

直接初始化:

建堆初始化:

二叉树的头删:

二叉树的尾插:

二叉树的取顶端元素:

二叉树的判空:

二叉树的销毁:

完整代码:


顺序存储:

简介:

顺序结构存储就是使用数组来存储 ,一般使用数组只适合表示完全二叉树 ,因为不是完全二叉树会有空间的浪费。而现实中使用中只有堆才会使用数组来存储,关于堆我们后面的章节会专门讲解。二叉树顺序存储在物理上是一个数组,在逻辑上是一颗二叉树。

节点的位置关系:

对于任意节点,设其在数组中的索引为 i(索引从0开始),则:

  • 父节点的索引为 (i-1)/2(向下取整)
  • 左子节点的索引为 2*i + 1
  • 右子节点的索引为 2*i + 2

优缺点:

优点

  • 访问任意节点的时间复杂度为O(1),因为可以直接通过索引访问。
  • 对于完全二叉树和满二叉树,空间利用率高。

缺点

  • 适用于完全二叉树或满二叉树,对于一般的二叉树,存在大量的空间浪费。
  • 插入和删除操作复杂,需要移动大量节点以保持树的顺序存储结构。

二叉树顺序存储的模拟实现:

向上调整算法:

cpp 复制代码
void AdjustUp(HeapDataType* 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;
		}
	}
}

向下调整算法:

cpp 复制代码
void Adjustdown(HeapDataType* a, int n, int parent)
{
	size_t child = parent * 2 + 1;
	while (child < n)
	{
		if (child + 1 < n && a[child + 1] < a[child])
		{
			child++;
		}
		if (a[child] > a[parent])
		{
			Swap(&a[child], &a[parent]);
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;
		}
	}
}

二叉树的初始化:

直接初始化:

cpp 复制代码
void HeapInit(HP* php)
{
	php->a = NULL;
	php->capacity = 0;
	php->size = 0;
}

建堆初始化:

cpp 复制代码
void HeapInitArrey(HP* php, HeapDataType* a, int n)
{
	assert(php);
	php->a = (HeapDataType*)malloc(sizeof(HeapDataType) * n);
	if (php->a == NULL)
	{
		perror("malloc fail:");
		return;
	}
	memcpy(php->a, a, sizeof(HeapDataType) * n);
}

二叉树的头删:

cpp 复制代码
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);
}

二叉树的尾插:

cpp 复制代码
void HeapPush(HP* php, HeapDataType x)
{
	if (php->size == php->capacity)
	{
		size_t newcapacity = php->capacity == 0 ? 4 : 2 * php->capacity;
		HeapDataType* tmp = realloc(php->a, sizeof(HeapDataType) * newcapacity);
		php->a = tmp;
		php->capacity = newcapacity;
	}
	php->a[php->size] = x;
	php->size++;

	AdjustUp(php->a, php->size - 1);
}

二叉树的取顶端元素:

cpp 复制代码
HeapDataType HeapTop(HP* php)
{
	assert(php);
	return php->a[0];
}

二叉树的判空:

cpp 复制代码
bool Empty(HP* php)
{
	assert(php);
	return php->size == 0;
}

二叉树的销毁:

cpp 复制代码
void HeapDestory(HP* php)
{
	free(php);
	php->a = NULL;
	php->capacity = 0; 
	php->size = 0;
}

完整代码:

cpp 复制代码
#include "heap.h"
void HeapInit(HP* php)
{
	php->a = NULL;
	php->capacity = 0;
	php->size = 0;
}

void HeapInitArrey(HP* php, HeapDataType* a, int n)
{
	assert(php);
	php->a = (HeapDataType*)malloc(sizeof(HeapDataType) * n);
	if (php->a == NULL)
	{
		perror("malloc fail:");
		return;
	}
	memcpy(php->a, a, sizeof(HeapDataType) * n);
}

void HeapPush(HP* php, HeapDataType x)
{
	if (php->size == php->capacity)
	{
		size_t newcapacity = php->capacity == 0 ? 4 : 2 * php->capacity;
		HeapDataType* tmp = realloc(php->a,sizeof(HeapDataType) * newcapacity);
		php->a = tmp;
		php->capacity = newcapacity;
	}
	php->a[php->size] = x;
	php->size++;

	AdjustUp(php->a, php->size - 1);
}

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);
}

void Swap(HeapDataType* x, HeapDataType* y)
{
	HeapDataType* tmp = *x;
	*x = *y;
	*y = tmp;
}

void AdjustUp(HeapDataType* 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;
		}
	}
}

void Adjustdown(HeapDataType* a, int n, int parent)
{
	size_t child = parent * 2 + 1;
	while (child < n)
	{
		if (child + 1 < n && a[child + 1] < a[child])
		{
			child++;
		}
		if (a[child] > a[parent])
		{
			Swap(&a[child], &a[parent]);
			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;
		}
	}
}

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

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

void HeapDestory(HP* php)
{
	free(php);
	php->a = NULL;
	php->capacity = 0; 
	php->size = 0;
}
相关推荐
顶呱呱程序5 分钟前
2-143 基于matlab-GUI的脉冲响应不变法实现音频滤波功能
算法·matlab·音视频·matlab-gui·音频滤波·脉冲响应不变法
&岁月不待人&12 分钟前
Kotlin by lazy和lateinit的使用及区别
android·开发语言·kotlin
StayInLove16 分钟前
G1垃圾回收器日志详解
java·开发语言
TeYiToKu17 分钟前
笔记整理—linux驱动开发部分(9)framebuffer驱动框架
linux·c语言·arm开发·驱动开发·笔记·嵌入式硬件·arm
无尽的大道23 分钟前
Java字符串深度解析:String的实现、常量池与性能优化
java·开发语言·性能优化
互联网打工人no124 分钟前
每日一题——第一百二十四题
c语言
爱吃生蚝的于勒27 分钟前
深入学习指针(5)!!!!!!!!!!!!!!!
c语言·开发语言·数据结构·学习·计算机网络·算法
羊小猪~~31 分钟前
数据结构C语言描述2(图文结合)--有头单链表,无头单链表(两种方法),链表反转、有序链表构建、排序等操作,考研可看
c语言·数据结构·c++·考研·算法·链表·visual studio
binishuaio36 分钟前
Java 第11天 (git版本控制器基础用法)
java·开发语言·git
zz.YE38 分钟前
【Java SE】StringBuffer
java·开发语言