当前位置: 首页 > news >正文

堆排序的插入和删除

插入:

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

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 一文读懂推荐系统
  • vue3使用elementui-plus时使用深度选择器穿透影响原样式
  • keepalived详解
  • Spring Mybatis拦截器配合logback打印完整sql语句
  • 嵌入式AI快速入门课程-K510篇 (第七篇 系统BSP开发)
  • GO语言如何抗住火影忍者手游的高并发
  • 入门网络安全工程师要学习哪些内容
  • 国自然放榜在即!用这种方法或可抢先查询...
  • 全新分支版本!微软推出Windows 11 Canary Build 27686版
  • ThreeJs学习笔记--坐标系,光源,相机控件
  • C/C++实现蓝屏2.0
  • Linux进程间通信——软件实现临界区互斥的基本方法
  • 高性能web服务器3——Nginx编译安装
  • Spring MVC Controller返回json日期格式配置失效的解决办法
  • CUDA编程07 - 卷积的优化
  • 2017 年终总结 —— 在路上
  • css布局,左右固定中间自适应实现
  • github指令
  • Java Agent 学习笔记
  • Java 内存分配及垃圾回收机制初探
  • JSONP原理
  • Redis中的lru算法实现
  • springboot_database项目介绍
  • tensorflow学习笔记3——MNIST应用篇
  • Theano - 导数
  • vagrant 添加本地 box 安装 laravel homestead
  • Web Storage相关
  • yii2中session跨域名的问题
  • 当SetTimeout遇到了字符串
  • 关于springcloud Gateway中的限流
  • 官方解决所有 npm 全局安装权限问题
  • 猫头鹰的深夜翻译:JDK9 NotNullOrElse方法
  • 容器服务kubernetes弹性伸缩高级用法
  • 使用 Xcode 的 Target 区分开发和生产环境
  • 线上 python http server profile 实践
  • kubernetes资源对象--ingress
  • 积累各种好的链接
  • 京东物流联手山西图灵打造智能供应链,让阅读更有趣 ...
  • ​【经验分享】微机原理、指令判断、判断指令是否正确判断指令是否正确​
  • ​LeetCode解法汇总307. 区域和检索 - 数组可修改
  • # Python csv、xlsx、json、二进制(MP3) 文件读写基本使用
  • # 安徽锐锋科技IDMS系统简介
  • #Lua:Lua调用C++生成的DLL库
  • $$$$GB2312-80区位编码表$$$$
  • $L^p$ 调和函数恒为零
  • (2024.6.23)最新版MAVEN的安装和配置教程(超详细)
  • (ISPRS,2023)深度语义-视觉对齐用于zero-shot遥感图像场景分类
  • (Redis使用系列) Springboot 在redis中使用BloomFilter布隆过滤器机制 六
  • (第61天)多租户架构(CDB/PDB)
  • (第一天)包装对象、作用域、创建对象
  • (二十六)Java 数据结构
  • (附源码)springboot金融新闻信息服务系统 毕业设计651450
  • (十八)Flink CEP 详解
  • (一)Kafka 安全之使用 SASL 进行身份验证 —— JAAS 配置、SASL 配置
  • (转)树状数组