堆排序的插入和删除

2024-08-21 16:44
文章标签 删除 插入 堆排序

本文主要是介绍堆排序的插入和删除,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

插入:

        1.  检查你的顺序表是否还有位置去插入,如果没有需要扩展

        2. 插入到已有序列的后一位置

        3. 和其父节点进行比较,是否满足大根堆/小根堆规则

        4. 不满足则需要交换数值

删除:

        1. 将最后一个元素覆盖将要删除的元素,然后将最后一个元素置空

图1-1输出说明:

        1.  建立大根堆然后排序结果

        2. 插入元素后 结果

        3. 插入元素 调整好新元素位置后 输出结果

        4. 重新排序结果

        5. 删除数据后结果

 

#include<stdio.h>
#include<stdlib.h>
#include<stdbool.h>/*大根堆:根节点的值大于左右孩子的数值*/
/*
1. 建立大根堆(检查所有的非终端结点 i<=n/2的结点  下标0舍弃不用)
2. 如果根节点不满足大根堆 那么将根节点与他最大的孩子结点交换 小元素不断下坠
3. 然后交换根节点和最后一个元素  根节点是最大的 最后一个元素是最小的
*/typedef struct {int* e;//存储堆数据int len;//堆的长度  元素个数int mlen;//堆的最大长度
}Heap;void InitHeap(Heap *hp,int n) {(*hp).e = (int*)malloc(sizeof(int) * (n + 1)); //0号位置不使用(*hp).len = 0;(*hp).mlen = n;
}//将数据填入数组中
void AddData(Heap hp, int d[], int n) {for (int i = 1; i <= n; i++) {hp.e[i] = d[i - 1];}
}
//将以k为根结点的树调整为大根堆
void HeapAdjust(Heap hp, int k, int n) {hp.e[0] = hp.e[k];   //临时存储该根节点的数据//沿着数值较大的子节点向下筛选for (int i = 2 * k; i <= n; i *= 2) {//左孩子结点if (i < n && hp.e[i] < hp.e[i + 1])i++;//右孩子大if (hp.e[0] > hp.e[i])  break;else {hp.e[k] = hp.e[i];k = i;}}hp.e[k] = hp.e[0];
}
void HeapAdjustLittle(Heap hp, int k, int n) {hp.e[0] = hp.e[k];   //临时存储该根节点的数据//沿着数值较大的子节点向下筛选for (int i = 2 * k; i <= n; i *= 2) {//左孩子结点if (i < n && hp.e[i] > hp.e[i + 1])i++;//右孩子小if (hp.e[0] < hp.e[i])  break;else {hp.e[k] = hp.e[i];k = i;}}hp.e[k] = hp.e[0];
}
//建立大根堆
void BuildBigRootHeap(Heap hp, int n) {for (int i = n / 2; i >= 1; i--) {HeapAdjust(hp, i, n);}}
void Print(Heap hp, int n) {for (int i = 1; i <= n; i++) {printf("%d   ", hp.e[i]);}printf("\n");
}//堆排序:每一趟将堆顶元素加入到有序子序列与待排序序列的最后一个元素交换
//交换以后,len长度数值-1  然后在将待排序树调整为大根堆  小元素下坠的过程 然后每一趟选出一个最大的数值
void HeapSort(Heap hp, int n) {for (int i = n; i >= 1; i--) {int tmp = hp.e[i];hp.e[i] = hp.e[1];hp.e[1] = tmp;//交换最后一个元素和堆顶元素HeapAdjust(hp, 1, i - 1);}
}
//插入元素//1. 扩展空间
/*void ExpandSpase(Heap *hp,int n) {int* p = hp->e;hp->e = (int*)realloc(hp->e,sizeof(int)*(hp->mlen + n));hp->mlen += n;free(p);
}*/
void ExpandSpase(Heap* hp, int n) {hp->e = (int*)realloc(hp->e, sizeof(int) * (hp->mlen + n + 1));hp->mlen += n; // 更新最大长度
}
void swap(int *a,int *b) {int tmp = *a;*a = *b;*b = tmp;
}
/*
注意点:
1. 此时堆已经是有序的了,所以已经由无序大根堆变为了有序小根堆
2. 所以在判断是否交换数据的时候,判断条件是 孩子节点是否大于父亲结点 如果是 那么不交换  否则交换*/
void InsertElement(Heap *hp,int value) {if (hp->len == hp->mlen)ExpandSpase(hp,1);//扩展一个空间hp->len += 1;hp->e[hp->len] = value;printf("插入元素但未调整:\n");Print(*hp, hp->len);hp->e[0] = hp->e[hp->len]; //使用数组下标为0的暂时存储新元素 new element 然后开始和父亲结点进行对比/交换int k = hp->len;//暂时存储孩子结点for (int i = k / 2; i >= 1;i/=2) {if (hp->e[i] < hp->e[k]) break;//孩子结点是否大于父亲  是就不交换 仍然满足小根堆swap(&hp->e[k],&hp->e[i]);k = i;//需要检查的结点(new point)向上移动}printf("调整后:\n");Print(*hp, hp->len);printf("重新排序:\n");
}
//删除:删除就是将最后一个元素覆盖要删除的点,然后数组长度-1  如果删除最后一个结点,那么直接-1即可
//然后采用元素下坠的办法进行调整(大元素下坠)
void DeleteElement(Heap *hp,int i) {//i表示删除第几个元素hp->e[i] = hp->e[hp->len];hp->len--;HeapAdjustLittle(*hp,i,hp->len);printf("删除后:\n");Print(*hp,hp->len);
}
int main() {int data[] = { 53,17,78,9,45,65,87,32 };//数据元素int n = 8;Heap hp;InitHeap(&hp,n);AddData(hp,data,n);hp.len = n;BuildBigRootHeap(hp,n);HeapSort(hp,n);Print(hp, n);//插入元素InsertElement(&hp,1);//重新排序  每次重新排序都要基于大根堆BuildBigRootHeap(hp,hp.len);HeapSort(hp,hp.len);Print(hp,hp.len);//删除1元素DeleteElement(&hp,1);free(hp.e);return 0;
}

这篇关于堆排序的插入和删除的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1093712

相关文章

使用C#代码在PDF文档中添加、删除和替换图片

《使用C#代码在PDF文档中添加、删除和替换图片》在当今数字化文档处理场景中,动态操作PDF文档中的图像已成为企业级应用开发的核心需求之一,本文将介绍如何在.NET平台使用C#代码在PDF文档中添加、... 目录引言用C#添加图片到PDF文档用C#删除PDF文档中的图片用C#替换PDF文档中的图片引言在当

macOS无效Launchpad图标轻松删除的4 种实用方法

《macOS无效Launchpad图标轻松删除的4种实用方法》mac中不在appstore上下载的应用经常在删除后它的图标还残留在launchpad中,并且长按图标也不会出现删除符号,下面解决这个问... 在 MACOS 上,Launchpad(也就是「启动台」)是一个便捷的 App 启动工具。但有时候,应

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

MySQL INSERT语句实现当记录不存在时插入的几种方法

《MySQLINSERT语句实现当记录不存在时插入的几种方法》MySQL的INSERT语句是用于向数据库表中插入新记录的关键命令,下面:本文主要介绍MySQLINSERT语句实现当记录不存在时... 目录使用 INSERT IGNORE使用 ON DUPLICATE KEY UPDATE使用 REPLACE

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

SQL Server清除日志文件ERRORLOG和删除tempdb.mdf

《SQLServer清除日志文件ERRORLOG和删除tempdb.mdf》数据库再使用一段时间后,日志文件会增大,特别是在磁盘容量不足的情况下,更是需要缩减,以下为缩减方法:如果可以停止SQLSe... 目录缩减 ERRORLOG 文件(停止服务后)停止 SQL Server 服务:找到错误日志文件:删除

mysql删除无用用户的方法实现

《mysql删除无用用户的方法实现》本文主要介绍了mysql删除无用用户的方法实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 1、删除不用的账户(1) 查看当前已存在账户mysql> select user,host,pa

Jmeter如何向数据库批量插入数据

《Jmeter如何向数据库批量插入数据》:本文主要介绍Jmeter如何向数据库批量插入数据方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Jmeter向数据库批量插入数据Jmeter向mysql数据库中插入数据的入门操作接下来做一下各个元件的配置总结Jmete

MySQL InnoDB引擎ibdata文件损坏/删除后使用frm和ibd文件恢复数据

《MySQLInnoDB引擎ibdata文件损坏/删除后使用frm和ibd文件恢复数据》mysql的ibdata文件被误删、被恶意修改,没有从库和备份数据的情况下的数据恢复,不能保证数据库所有表数据... 参考:mysql Innodb表空间卸载、迁移、装载的使用方法注意!此方法只适用于innodb_fi