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

数据结构:堆和堆排序

数据结构:堆和堆排序

文章目录

  • 数据结构:堆和堆排序
    • 1.二叉树的存储结构
      • 1.顺序结构
      • 2.链式结构
    • 2.堆
    • 3.堆的实现
    • 4.堆排序(选择排序中的一类)
      • 1. 基本思想
      • 2.代码实现

1.二叉树的存储结构

1.顺序结构

顺序结构存储就是使用数组来表示一棵二叉树一般使用数组只适合表示完全二叉树,因为不是完全二叉树会有空间的浪费。而现实使用中只有堆才会使用数组来存储二叉树顺序存储在物理上是一个数组,在逻辑上是一颗二叉树。

2.链式结构

二叉树的链式存储是使用链表来表示一棵二叉树,即用指针链接来指示元素的逻辑关系。 通常的方法是
链表中每个结点由三个域组成,数据域和左右指针域,左右指针分别用来给出该结点左孩子和右孩子所
在的链结点的存储地址 。
在这里插入图片描述

2.堆

堆实际上就是一棵完全二叉树

其性质为:

  • 堆中某个节点的值总是不大于或不小于其父节点的值;
  • 堆总是一棵完全二叉树。

大堆和小堆:

  • 每个节点的值都大于或等于其子节点的值,为大堆;反之为小堆。

3.堆的实现

#define _CRT_SECURE_NO_WARNINGS 1#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include<stdbool.h>
// (大小)堆的实现(堆的结构与特点发现与数组下标有紧密关系)因此可以使用顺序表结构体
typedef int HPDataType;
typedef struct Heap
{HPDataType* a;int size;int capacity;
}Heap;// 堆的构建
void HeapCreate(Heap* hp);// 堆的销毁
void HeapDestory(Heap* hp);
// 向上调整
void AdjustUp(HPDataType* a, size_t childposition);// 堆的插入
void HeapPush(Heap* hp, HPDataType x);
// 向下调整
void AdjustDown(HPDataType* a, int size, size_t parentposition);// 堆的删除
void HeapPop(Heap* hp);// 取堆顶的数据
HPDataType HeapTop(Heap* hp);// 堆的数据个数
int HeapSize(Heap* hp);// 堆的判空
bool HeapEmpty(Heap* hp);

#include "Heap.h"// 堆的构建
void HeapCreate(Heap* hp)
{assert(hp);hp->a = NULL;hp->capacity = 0;hp->size = 0;
}
// 堆的销毁
void HeapDestory(Heap* hp)
{free(hp->a);hp->capacity = 0;hp->size = 0;hp = NULL;
}
// 向上调整
void AdjustUp(HPDataType* a, size_t childposition)
{assert(a);size_t parentposition = (childposition - 1) / 2;while (childposition > 0){// 小堆if (a[childposition] < a[parentposition]){// 交换孩子和父亲的值HPDataType  tmp = a[childposition];a[childposition] = a[parentposition];a[parentposition] = tmp;childposition = parentposition;parentposition = (parentposition - 1) / 2;}else{return;}}
}
// 堆的插入
void HeapPush(Heap* hp, HPDataType x)
{assert(hp);// 检查容量if (hp->size == hp->capacity){int newcapacity = hp->capacity == 0 ? 4 : hp->capacity * 2;HPDataType* newa = (HPDataType*)realloc(hp->a, sizeof(HPDataType) * newcapacity);if (newa == NULL){perror("realloc fail");return;}hp->a = newa;hp->capacity = newcapacity;}hp->a[hp->size] = x;hp->size++;// 向上调整(可用递归可用循环,此处我们用循环实现)AdjustUp(hp->a, hp->size - 1);}
// 向下调整
void AdjustDown(HPDataType* a, int size, size_t parentposition)
{assert(a);size_t childposition = a[1] < a[2] ? 1 : 2;while (childposition < size){if (a[childposition] < a[parentposition])// 小堆{HPDataType tmp = a[childposition];a[childposition] = a[parentposition];a[parentposition] = tmp;parentposition = childposition;childposition = a[childposition * 2 + 1] < a[childposition * 2 + 2] ? childposition * 2 + 1 : childposition * 2 + 2;}else{break;}}
}// 堆的删除(删除堆顶元素)
void HeapPop(Heap* hp)
{assert(hp);if (hp->size == 0){printf("堆为空,无法删除\n");return;}HPDataType tmp = hp->a[0];hp->a[0] = hp->a[hp->size - 1];hp->size--;AdjustDown(hp->a, hp->size, 0);}
// 取堆顶的数据
HPDataType HeapTop(Heap* hp)
{assert(hp);return hp->a[0];
}
// 堆的数据个数
int HeapSize(Heap* hp)
{assert(hp);return hp->size;
}
// 堆的判空
bool HeapEmpty(Heap* hp)
{assert(hp);if (hp->size == 0){return true;}else{return false;}
}

4.堆排序(选择排序中的一类)

看到堆排序我们会想到是不是利用我们自己造轮子写出来的堆这个数据结构和接口函数来实现排序呢?

假设我们要对一个数组里的元素进行排序,我们可能想到这样做:

  1. 将数组里的元素通通插入到堆中,插入进去的元素自然就形成了一种排序结构。
  2. 再将堆里的数据取出来放回到原数组中从而进行排序。

弊端:这样进行‘’堆排序‘’实际上是十分复杂和浪费空间和时间的,这个方法前提下我们必须要先自己实现一个堆,再然后堆的构建也是要开辟内存空间的十分低效。

一次建堆的时间的复杂度是(向上调整建堆和向下调整建堆):
O ( N ∗ log ⁡ N ) O(N*\log_{}{N}) O(NlogN)


实际的堆排序:

记忆:升序建大堆,降序建小堆。

1. 基本思想

利用大顶堆(小顶堆)堆顶记录的是最大关键字(最小关键字)这一特性,使得每次从无序中选择最大记录(最小记录)变得简单。

① 将待排序的序列构造成一个==最大堆==,此时序列的最大值为根节点。
② 依次将根节点与待排序序列的最后一个元素交换。
③ 再维护从根节点到该元素的前一个节点为最大堆,**如此往复,最终得到一个递增序列**。

2.代码实现

// 向上调整
void AdjustUp(int* a, size_t childposition)
{assert(a);size_t parentposition = (childposition - 1) / 2;while (childposition > 0){// 大堆if (a[childposition] > a[parentposition]){// 交换孩子和父亲的值int  tmp = a[childposition];a[childposition] = a[parentposition];a[parentposition] = tmp;childposition = parentposition;parentposition = (parentposition - 1) / 2;}else{return;}}
}void swap(int* start, int* end)
{int tmp = 0;tmp = *start;*start = *end;*end = tmp;
}// 堆排序
void HeapSort(int* a, int n)
{int i = 0;// 当数据个数大于1时才进行排序,否则直接就是有序的while (n > 1){// 建大堆使得最大的数位于堆顶部for (i = 1; i < n; i++){AdjustUp(a, i);}// 堆顶元素与堆最后一个元素交换位置swap(&a[0], &a[n - 1]);// 除去堆最后一个位置的元素重新建立大堆,如此往复,最终得到一个递增序列n--;}
}

#define  _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>void HeapSort(int* a, int n);
void AdjustUp(int* a, size_t childposition);int main()
{int a[] = { 4,6,2,1,5,8,9,3,7,10 };HeapSort(a, sizeof(a) / sizeof(a[0]));int i = 0;for (i = 0; i < sizeof(a) / sizeof(a[0]); i++){printf("%d ", a[i]);}printf("\n");return 0;
}

在这里插入图片描述

在这里插入图片描述

相关文章:

数据结构:堆和堆排序

数据结构&#xff1a;堆和堆排序 文章目录 数据结构&#xff1a;堆和堆排序1.二叉树的存储结构1.顺序结构2.链式结构 2.堆3.堆的实现4.堆排序&#xff08;选择排序中的一类&#xff09;1. 基本思想2.代码实现 1.二叉树的存储结构 1.顺序结构 顺序结构存储就是使用数组来表示一…...

力扣精选算法100题——水果成篮(滑动窗口专题)

本题链接&#x1f449;水果成篮 第一步&#xff1a;了解题意 我就按照实例1来进行对这题的理解。 1代表种类类型&#xff0c;这个数组里面有2个种类类型 ps:种类1和种类2 &#xff0c;只不过种类1是有2个水果&#xff0c;种类2有一个水果&#xff0c;共计3个水果。 本题需要解…...

【提示学习论文六】MaPLe: Multi-modal Prompt Learning论文原理

文章目录 MaPLe: Multi-modal Prompt Learning 多模式提示学习文章介绍动机MaPLe:Multi-modal Prompt Learning 模型结构1、Deep Language Prompting 深度语言提示2、Deep Vision Prompting 深度视觉提示3、Vision Language Prompt Coupling 视觉语言提示耦合提示耦合过程 实验…...

wpf使用Popup封装数据筛选框

(关注博主后,在“粉丝专栏”,可免费阅读此文) 类似于DevExpress控件的功能 这是DevExpress的winform筛选样式,如下: 这是DevExpress的wpf筛选样式,如下: 这是Excel的筛选样式,如下: 先看效果 本案例使用wpf原生控件封装,功能基本上都满足,只是颜色样式没有写…...

微信小程序 - 视图与逻辑 介绍

文章目录 视图与逻辑一、页面导航1、页面导航 - 声明式导航1.1 导航到tabBar页面1.2 导航到非tabBar页面1.3 后退导航 2、页面导航 - 编程式导航2.1 导航到tabBar页面2.2 导航到非tabBar页面2.3 后退导航 3、页面导航 - 导航传参3.1 声明式导航传参3.2 编程式导航传参3.3 在 on…...

大创项目推荐 深度学习疫情社交安全距离检测算法 - python opencv cnn

文章目录 0 前言1 课题背景2 实现效果3 相关技术3.1 YOLOV43.2 基于 DeepSort 算法的行人跟踪 4 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; **基于深度学习疫情社交安全距离检测算法 ** 该项目较为新颖&#xff0c;适合作为竞赛…...

c语言-库函数strstr()、strtok()、strerror()介绍

文章目录 前言一、库函数strstr()1.1 strstr()介绍1.2 strstr()模拟实现 二、库函数strtok()2.1 strtok()介绍 三、库函数strerror()3.1 strerror()介绍 总结 前言 本篇文章介绍c语言库函数strstr()、strtok()、strerror()的使用。 一、库函数strstr() 1.1 strstr()介绍 str…...

RIP【新华三与华为区别】

【介绍】 rip分为rip 1 与 rip 2 &#xff0c;rip 2 是对 rip 1 的一种升级&#xff0c;rip 2 可以进行认证等功能 【命令】 新华三&#xff1a; [HC3-R1] rip #启用rip [HC3-R1-rip] version 2 #告知rip 版本号 [HC3-R1-rip] network 192.168.1.0 #宣告其网段 [HC3-R1-rip] …...

Python从入门到精通秘籍四

Python速成,知识点超详细,跟着这个系列边输入边学习体会吧! 一、Python的判断语句的综合案例 下面是一个使用代码示例来详细说明Python判断语句的综合案例,通过用户输入来实现简单的登录验证: # 提示用户输入用户名和密码 username = input("请输入用户名:")…...

rk3568下SoftBusDumpDeviceInfo执行错误—鸿蒙开发已解决

文章目录 项目场景:问题描述原因分析:解决方案:此Bug解决方案总结寄语项目场景: 最近也是遇到了这个问题,看到网上也有人在询问这个问题,本文总结了自己和其他人的解决经验,解决了rk3568下SoftBusDumpDeviceInfo执行错误的问题。 命令行运行 SoftBusDumpDeviceInfo,测…...

Vue 3 Composition API 详解

一、引言 在Vue 3中&#xff0c;引入了一个新的Composition API&#xff0c;旨在提供一种更灵活和可重用的方式来组织组件代码。Composition API基于函数式编程思想&#xff0c;允许开发者将逻辑和状态管理逻辑分离&#xff0c;使代码更加清晰和可维护。 二、Composition API…...

API设计:从基础到最佳实践

1*vWvkkgG6uvgmJT8GkId98A.png 在这次深入探讨中&#xff0c;我们将深入了解API设计&#xff0c;从基础知识开始&#xff0c;逐步进阶到定义出色API的最佳实践。 作为开发者&#xff0c;你可能对许多这些概念很熟悉&#xff0c;但我将提供详细的解释&#xff0c;以加深你的理解…...

每日汇评:由于中东危机削弱了风险偏好,欧元将在1.0900附近波动

随着中东危机的加深&#xff0c;欧元兑美元面临大幅抛售&#xff1b; 由于高通胀&#xff0c;欧洲央行决策者推迟了市场对早期降息的预期&#xff1b; 市场将受到周三公布的美国零售销售数据的影响&#xff1b; 持续的中东紧张局势增强了对避险资产的吸引力&#xff0c;而风险感…...

算法每日一题:删除子串后的字符串最小长度 | 栈 | 字符串

大家好&#xff0c;我是星恒 今天给大家带来的是一道另类的栈的应用 话不多说&#xff0c;我们直接来体验 题目&#xff1a;leetcode 2696 给你一个仅由 大写 英文字符组成的字符串 s 。你可以对此字符串执行一些操作&#xff0c;在每一步操作中&#xff0c;你可以从 s 中删除 …...

SpringFramework实战指南(一)

SpringFramework实战指南&#xff08;一&#xff09; 一、技术体系结构1.1 总体技术体系1.2 框架概念和理解 一、技术体系结构 1.1 总体技术体系 单一架构 一个项目&#xff0c;一个工程&#xff0c;导出为一个war包&#xff0c;在一个Tomcat上运行。也叫all in one。 单一架…...

AtCoder ABC198

本期F为群论题&#xff0c;很有难度。 C - Compass Walking 为了避免精度问题&#xff0c;采用二分推算。但是要小心结果为1的地方。 R 2 ∗ k 2 ≥ x 2 y 2 R^2*k^2\geq x^2y^2 R2∗k2≥x2y2 # -*- coding: utf-8 -*- # time : 2023/6/2 13:30 # file : atcoder.…...

phpinfo和php -m 加载的php.ini不一致

目的&#xff1a; 将phpinfo在web中展示的php.ini和在命令行中展示的php.ini加载路径设置一致。 原本的php.ini加载路劲是&#xff1a; /usr/local/lib/php.ini 解决思路&#xff1a; &#xff08;1&#xff09;which php 查看服务器加载的php的位置&#xff0c;这里原来是&a…...

121.买卖股票的最佳时机 122.买卖股票的最佳时机II

121.买卖股票的最佳时机 122.买卖股票的最佳时机II 121.买卖股票的最佳时机 力扣题目链接(opens new window) 给定一个数组 prices &#xff0c;它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。 你只能选择 某一天 买入这只股票&#xff0c;并选择在 未来的某一…...

Spring Boot整理-Spring Boot是什么?

Spring Boot 是一个开源的 Java 基础框架,它旨在简化基于 Spring 的应用开发。其核心特点在于“约定优于配置”的设计哲学,意味着它提供了一系列默认配置,从而帮助开发者更快地启动和运行新的 Spring 应用。Spring Boot 的主要特点包括: 自动配置:Spring Boot 可以根据项目…...

【pytorch】Pytorch 中的 grid 与 各种变换

Pytorch 中的 grid 与 各种变换 数学原理 **单应性&#xff08;Homography&#xff09; : 也就是透视变换。**单应性最初用来研究欧几里得几何中的透视和投影&#xff0c;而单应性一词&#xff0c;从词源学上来说&#xff0c;大致意思是“相似的绘图”。单应性的概念被引入来…...

如何在现代显示器上完美重温经典游戏?终极宽屏修复工具包指南

如何在现代显示器上完美重温经典游戏&#xff1f;终极宽屏修复工具包指南 【免费下载链接】WidescreenFixesPack Plugins to make or improve widescreen resolutions support in games, add more features and fix bugs. 项目地址: https://gitcode.com/gh_mirrors/wi/Wides…...

Burp Suite渗透测试工作流:从环境搭建到报告生成

1. 这不是“学个工具”&#xff0c;而是一套可复用的渗透工作流很多人点开“Burp Suite 入门”类教程&#xff0c;心里想的是&#xff1a;“装个插件、抓个包、改个参数&#xff0c;不就完事了&#xff1f;”——结果三天后连 repeater 怎么发 POST 请求都得翻笔记。我带过二十…...

告别断电重启就丢程序:深入聊聊紫光同创FPGA的Flash固化与CPLD内置eFlash配置差异

紫光同创FPGA与CPLD配置存储机制深度解析&#xff1a;从瞬态下载到永久固化的技术实现 在数字电路设计领域&#xff0c;FPGA和CPLD的可重构特性为硬件开发带来了极大灵活性。然而&#xff0c;这种灵活性背后需要可靠的配置存储机制作为支撑——断电后程序能否自动恢复&#xf…...

生成式引擎优化的技术底座:JSON-LD 结构化数据标记全指南

为什么你的内容 AI 搜索"读不懂" 生成式引擎优化&#xff08;GEO&#xff09;已经不是什么新概念了。信通院在2026年5月发布的《生成式引擎优化&#xff08;GEO&#xff09;白皮书》中指出&#xff0c;超过60%的企业内容未被 AI 搜索引擎正确理解和引用&#xff0c;…...

Lamini:5分钟快速搭建专属AI模型的高效Python客户端

Lamini&#xff1a;5分钟快速搭建专属AI模型的高效Python客户端 【免费下载链接】lamini The Official Python Client for Laminis API 项目地址: https://gitcode.com/gh_mirrors/la/lamini Lamini作为一款革命性的AI开发平台&#xff0c;为技术开发者和AI爱好者提供了…...

VideoDownloadHelper:打破网页视频下载壁垒的智能解决方案

VideoDownloadHelper&#xff1a;打破网页视频下载壁垒的智能解决方案 【免费下载链接】VideoDownloadHelper Chrome Extension to Help Download Video for Some Video Sites. 项目地址: https://gitcode.com/gh_mirrors/vi/VideoDownloadHelper 你是否曾遇到过这样的困…...

鸿蒙 PC:从“用户点击”到“AI 调度”

子玥酱 &#xff08;掘金 / 知乎 / CSDN / 简书 同名&#xff09; 大家好&#xff0c;我是 子玥酱&#xff0c;一名长期深耕在一线的前端程序媛 &#x1f469;‍&#x1f4bb;。曾就职于多家知名互联网大厂&#xff0c;目前在某国企负责前端软件研发相关工作&#xff0c;主要聚…...

如何用Happy Island Designer免费打造你的梦幻岛屿:终极完整指南

如何用Happy Island Designer免费打造你的梦幻岛屿&#xff1a;终极完整指南 【免费下载链接】HappyIslandDesigner "Happy Island Designer (Alpha)"&#xff0c;是一个在线工具&#xff0c;它允许用户设计和定制自己的岛屿。这个工具是受游戏《动物森友会》(Animal…...

RoPE与KV缓存优化:提升Transformer长序列处理能力

1. 旋转位置编码&#xff08;RoPE&#xff09;技术解析旋转位置编码&#xff08;Rotary Position Embedding, RoPE&#xff09;是近年来Transformer架构中位置编码技术的重要突破。传统Transformer使用绝对或相对位置编码&#xff0c;而RoPE通过旋转矩阵实现位置信息的注入&…...

物理学巅峰成就巡礼:从牛顿到量子,探索宇宙与微观世界的革命性突破

1. 项目概述&#xff1a;一次对物理学巅峰成就的巡礼2019年&#xff0c;诺贝尔物理学奖授予了三位天体物理学家——詹姆斯皮布尔斯、米歇尔马约尔和迪迪埃奎洛兹&#xff0c;以表彰他们在物理宇宙学理论以及系外行星发现领域的开创性贡献。这个奖项像一束聚光灯&#xff0c;将公…...