堆(顺序结构)

一、概念

树是⼀种⾮线性的数据结构,它是由n(n>=0) 个有限结点组成⼀个具有层次关系的集合。把它叫做树是因为它看起来像⼀棵倒挂的树,也就是说它是根朝上,⽽叶朝下的

数是有一个根节点和其余排列有序的子节点组成的

结点的度:⼀个结点有⼏个孩⼦,他的度就是多少;

树的度:⼀棵树中,最⼤的结点的度称为树的度;

结点的层次:从根开始定义起,根为第1 层,根的⼦结点为第2层,以此类推;

树的⾼度或深度:树中结点的最⼤层次;

二、二叉树

在树形结构中,我们最常⽤的就是⼆叉树,⼀棵⼆叉树是结点的⼀个有限集合,该集合由⼀个根结点加上两棵别称为左⼦树和右⼦树的⼆叉树组成或者为空

1. ⼆叉树不存在度⼤于2的结点
2. ⼆叉树的⼦树有左右之分,次序不能颠倒,因此⼆叉树是有序树

满二叉树

⼆叉树的层数为K,且结点总数是2^k-1;

完全二叉树

对于深度为K的,有n个结点的⼆叉树,当且仅当其每⼀个结点都与深度为K的满⼆叉树中编号从1⾄n的结点⼀⼀对应时称之为完全⼆叉树

完全二叉树包含满二叉树

链式结构和顺序结构

1、顺序结构就是用数组来存储数据

2、链式结构

分两种

三、堆

1、概念

这里的堆不是栈区堆区中的概念,这里是指将一个集合的所有元素按完全⼆叉树的顺序存储⽅式存储

小堆:堆顶的元素是最小的,按照堆的方式将元素依次增大排列下去,只要保证每个根节点是小于它的字节点

大堆:堆顶的元素是最大的,按照堆的方式将元素依次减小排列下去
假设一个位置的节点序号为i,则他的双亲序号为(i-1)/2 ;i=0 ,i为根结点编号,⽆双亲结点;则他的左右孩子为2i+1,2i+2
堆的本质结构还是数组,因此其底层代码是数组

c 复制代码
typedef int HPDataType;

typedef struct Heap
{
	HPDataType* arr;
	int size;      //有效数据个数
	int capacity;  //空间大小
}HP;

//堆的初始化
void HPInit(HP* php)
{
	assert(php);
	php->arr = NULL;
	php->size = php->capacity = 0;
}

//堆的销毁
void HPDestroy(HP* php)
{
	assert(php);
	if (php->arr)
		free(php->arr);
	php->arr = NULL;
	php->size = php->capacity = 0;
}

//堆的打印
void HPPrint(HP* php)
{
	for (int i = 0; i < php->size; i++)
	{
		printf("%d ", php->arr[i]);
	}
	printf("\n");
}


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

//向上调整,拿孩子跟父亲进行比较
void AdjustUp(HPDataType* arr, int child) {
	int parent = (child - 1) / 2;
	while (child > 0) {
		//小堆:  <
		//大堆:   >
		if (arr[child] > arr[parent])
		{
			Swap(&arr[child], &arr[parent]);
			child = parent;
			parent = (child - 1) / 2;
		}
		else {
			break;
		}
	}
}

//向下调整
void AdjustDown(HPDataType* arr, int parent, int n) {
	int child = parent * 2 + 1;
	while (child < n)
	{
		//大堆: <
		//小堆: >
		if (child + 1 < n && arr[child] < arr[child + 1])
		{
			child++;
		}
		//大堆:>
		//小堆: <
		if (arr[child] > arr[parent])
		{
			Swap(&arr[child], &arr[parent]);
			parent = child;
			child = parent * 2 + 1;
		}
		else {
			break;
		}
	}
}

//将元素压入堆中
void HPPush(HP* php, HPDataType x) {
	assert(php);
	//判断空间足够
	if (php->size == php->capacity)
	{
		//增容
		int newCapacity = php->capacity == 0 ? 4 : 2 * php->capacity;
		HPDataType* tmp = (HPDataType*)realloc(php->arr, newCapacity * sizeof(HPDataType));
		if (tmp == NULL)
		{
			perror("realloc fail!");
			exit(1);
		}
		php->arr = tmp;
		php->capacity = newCapacity;
	}
	//直接插入
	php->arr[php->size] = x;
	//向上调整
	AdjustUp(php->arr, php->size);
	++php->size;
}

//判空
bool HPEmpty(HP* php)
{
	assert(php);
	return php->size == 0;
}

//出堆
void HPPop(HP* php)
{
	assert(!HPEmpty(php));
	//0  php->size-1
	Swap(&php->arr[0], &php->arr[php->size - 1]);
	--php->size;
	//向下调整
	AdjustDown(php->arr, 0, php->size);
}

//返回堆顶元素
HPDataType HPTop(HP* php)
{
	assert(!HPEmpty(php));
	return php->arr[0];
}


向上调整的时间复杂度为:n*log 2 (n)

向下调整的时间复杂度为:n

四、堆的应用

堆排序

c 复制代码
//时间复杂度:n*log(n)
void HeapSort(int* arr, int n)
{
	//向下建堆
	//for (int i = (n-1-1)/2; i >= 0; i--)
	//{
	//	AdjustDown(arr, i, n);
	//}

	// 建堆---向上调整
	for (int i = 0; i < n; i++)
	{
		AdjustUp(arr, i);
	}

	//堆排序
	//使用大堆来实现升序(就是让元素按数组的方式从小到大排列)
	int end = n - 1;
	while (end > 0)
	{
		Swap(&arr[0], &arr[end]);
		AdjustDown(arr, 0, end);
		end--;
	}
}

尽量用向下建堆的方法,时间复杂度更小

五、TOP-K问题

TOP-K问题:即求数据结合中前K个最⼤的元素或者最⼩的元素,⼀般情况下数据量都⽐较⼤

前k个最⼤的元素,则建⼩堆

前k个最⼩的元素,则建⼤堆

c 复制代码
//TOPK
void CreateNDate()
{
	// 造数据
	int n = 100000;
	srand(time(0));
	const char* file = "data.txt";
	FILE* fin = fopen(file, "w");
	if (fin == NULL)
	{
		perror("fopen error");
		return;
	}
	for (int i = 0; i < n; ++i)
	{
		int x = (rand() + i) % 1000000;
		fprintf(fin, "%d\n", x);
	}
	fclose(fin);
}


void Topk()
{
	int k = 0;
	printf("请输入K:");
	scanf("%d", &k);

	const char* file = "data.txt";
	FILE* fout = fopen(file, "r");
	if (fout == NULL)
	{
		perror("fopen error");
		exit(1);
	}

	//找最大的前K个数据,建小堆
	int* minHeap = (int*)malloc(sizeof(int) * k);
	if (minHeap == NULL)
	{
		perror("malloc fail!");
		exit(2);
	}
	for (int i = 0; i < k; i++)
	{
		fscanf(fout, "%d", &minHeap[i]);
	}
	//建堆--向下调整建堆
	for (int i = (k - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDown(minHeap, i, k);
	}
	//遍历剩下的n-k个数,跟堆顶进行比较,谁大谁入堆
	//调整堆
	int x = 0;
	while (fscanf(fout,"%d",&x) != EOF)
	{
		if (x > minHeap[0])
		{
			minHeap[0] = x;
			AdjustDown(minHeap, 0, k);
		}
	}

	for (int i = 0; i < k; i++)
	{
		printf("%d ", minHeap[i]);
	}

	fclose(fout);
}
相关推荐
宫瑾2 小时前
VSCode使用C/C++ extensions开发STM32,添加头文件路径
c语言·c++·vscode
m0_531237172 小时前
C语言-编程实例
c语言·开发语言·数据结构
Felven3 小时前
C. Yet Another Card Deck
c语言·开发语言
小龙报3 小时前
【算法通关指南:数据结构与算法篇】二叉树相关算法题:1.美国血统 American Heritage 2.二叉树问题
c语言·数据结构·c++·算法·深度优先·广度优先·宽度优先
无限进步_5 小时前
面试题 02.04. 分割链表 - 题解与详细分析
c语言·开发语言·数据结构·git·链表·github·visual studio
Once_day17 小时前
C++之《程序员自我修养》读书总结(4)
c语言·c++·编译和链接
三无少女指南17 小时前
开发者环境配置:用 Ollama 实现本地大模型部署(附下载慢的解决方案
c语言·开发语言·数据库·ubuntu
2501_9181269118 小时前
stm32核心板是什么属性?
linux·c语言·stm32·嵌入式硬件·个人开发
m0_5312371718 小时前
C语言-操作符练习
c语言·开发语言