堆排序的插入和删除

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

相关文章

docker如何删除悬空镜像

《docker如何删除悬空镜像》文章介绍了如何使用Docker命令删除悬空镜像,以提高服务器空间利用率,通过使用dockerimage命令结合filter和awk工具,可以过滤出没有Tag的镜像,并将... 目录docChina编程ker删除悬空镜像前言悬空镜像docker官方提供的方式自定义方式总结docker

使用Python在Excel中插入、修改、提取和删除超链接

《使用Python在Excel中插入、修改、提取和删除超链接》超链接是Excel中的常用功能,通过点击超链接可以快速跳转到外部网站、本地文件或工作表中的特定单元格,有效提升数据访问的效率和用户体验,这... 目录引言使用工具python在Excel中插入超链接Python修改Excel中的超链接Python

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

C#实现添加/替换/提取或删除Excel中的图片

《C#实现添加/替换/提取或删除Excel中的图片》在Excel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更加美观,下面我们来看看如何在C#中实现添加/替换/提取或删除E... 在Excandroidel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Git中恢复已删除分支的几种方法

《Git中恢复已删除分支的几种方法》:本文主要介绍在Git中恢复已删除分支的几种方法,包括查找提交记录、恢复分支、推送恢复的分支等步骤,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录1. 恢复本地删除的分支场景方法2. 恢复远程删除的分支场景方法3. 恢复未推送的本地删除分支场景方法4. 恢复

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

Oracle数据库使用 listagg去重删除重复数据的方法汇总

《Oracle数据库使用listagg去重删除重复数据的方法汇总》文章介绍了在Oracle数据库中使用LISTAGG和XMLAGG函数进行字符串聚合并去重的方法,包括去重聚合、使用XML解析和CLO... 目录案例表第一种:使用wm_concat() + distinct去重聚合第二种:使用listagg,

Redis过期键删除策略解读

《Redis过期键删除策略解读》Redis通过惰性删除策略和定期删除策略来管理过期键,惰性删除策略在键被访问时检查是否过期并删除,节省CPU开销但可能导致过期键滞留,定期删除策略定期扫描并删除过期键,... 目录1.Redis使用两种不同的策略来删除过期键,分别是惰性删除策略和定期删除策略1.1惰性删除策略

SpringBoot项目删除Bean或者不加载Bean的问题解决

《SpringBoot项目删除Bean或者不加载Bean的问题解决》文章介绍了在SpringBoot项目中如何使用@ComponentScan注解和自定义过滤器实现不加载某些Bean的方法,本文通过实... 使用@ComponentScan注解中的@ComponentScan.Filter标记不加载。@C