排序算法(六)--堆排序

文章目录

堆排序 C语言实例

引言

堆排序(Heap Sort)作为一种基于堆数据结构的比较排序算法,以其时间复杂度稳定、实现相对简单而备受青睐。

堆的基本概念

堆是一种特殊的完全二叉树,分为最大堆和最小堆。在最大堆中,每个节点的值都大于或等于其子节点的值;在最小堆中,每个节点的值都小于或等于其子节点的值。堆排序通常使用最大堆来实现升序排序,通过反复将堆顶元素(最大值)与堆末元素交换,并重新调整堆结构,逐步完成排序过程。

堆排序算法步骤
构建最大堆 :将待排序数组视为完全二叉树,通过"堆化"过程将其调整为最大堆。堆化是指从最后一个非叶子节点开始,向上至根节点,逐个调整节点位置,确保每个节点都满足最大堆性质。
交换堆顶与堆末元素 :将堆顶元素(当前最大值)与堆末元素交换,使最大值移动到数组末尾。此时,堆的有效范围缩小,堆末元素(已排序的最大值)不再参与堆调整。
重新调整堆:对剩余元素继续执行堆化操作,保持最大堆性质。重复步骤2和步骤3,直至堆的有效范围缩小至只剩一个元素,排序完成。

C语言实现

以下是堆排序的C语言实现代码,包括构建最大堆、堆化和堆排序的主要函数:

#include <stdio.h>

// 堆化函数,调整以i为根的子树为最大堆

void heapify(int arr\[\], int n, int i) {

int largest = i; // 初始化最大值为根节点

int left = 2 * i + 1; // 左子节点

int right = 2 * i + 2; // 右子节点

// 如果左子节点存在且大于根节点,则更新最大值

if (left < n && arrleft > arrlargest)

largest = left;

// 如果右子节点存在且大于当前最大值,则更新最大值

if (right < n && arrright > arrlargest)

largest = right;

// 如果最大值不是根节点,则交换并继续堆化

if (largest != i) {

int swap = arri;

arri = arrlargest;

arrlargest = swap;

heapify(arr, n, largest); // 递归堆化受影响的子树

}

}

// 堆排序函数

void heapSort(int arr\[\], int n) {

// 构建最大堆

for (int i = n / 2 - 1; i >= 0; i--)

heapify(arr, n, i);

// 逐个将堆顶元素与堆末元素交换,并调整剩余堆

for (int i = n - 1; i >= 0; i--) {

int swap = arr0;

arr0 = arri;

arri = swap;

heapify(arr, i, 0); // 对剩余元素重新堆化

}

}

// 打印数组函数

void printArray(int arr\[\], int size) {

for (int i = 0; i < size; i++)

printf("%d ", arri);

printf("\n");

}

// 主函数

int main() {

int arr\[\] = {12, 11, 13, 5, 6, 7};

int n = sizeof(arr) / sizeof(arr0);

printf("未排序数组: \n");

printArray(arr, n);

heapSort(arr, n);

printf("已排序数组: \n");

printArray(arr, n);

return 0;

}

代码解析

heapify函数 :负责调整以i为根的子树,确保满足最大堆性质。通过比较根节点、左子节点和右子节点的值,找到最大值并交换位置。如果最大值不是根节点,则递归调用heapify函数继续调整。
heapSort函数 :首先通过从最后一个非叶子节点开始向上遍历,调用heapify函数构建最大堆。然后,逐个将堆顶元素(最大值)与堆末元素交换,并减小堆的有效范围。对剩余元素继续调用heapify函数,保持最大堆性质,直至排序完成。
printArray函数 :用于打印数组内容,方便观察排序前后的变化。
main函数:定义待排序数组,调用heapSort函数进行排序,并打印排序前后的数组内容。

结论

堆排序作为一种高效的排序算法,通过构建最大堆和不断调整堆结构,实现了稳定的排序效果。

相关推荐
zfoo-framework6 分钟前
[修改代码使用]codex官方app中使用中转(不需要cc-switch) 1.config.toml 2.sk方式登录
java
逍遥德26 分钟前
MQTT教程详解-05.SpringBoot集成mqtt client 性能分析
java·spring boot·spring·mt
云烟成雨TD30 分钟前
Spring AI 1.x 系列【54】Retry 机制分析
java·人工智能·spring
weixin_5231853233 分钟前
Collections.unmodifiableMap详解:真的不可修改吗?
java·linux·前端
手写码匠33 分钟前
从零实现 Prompt 工程引擎:结构化提示、自动优化与多轮自省体系
人工智能·深度学习·算法·aigc
点燃大海34 分钟前
SpringAI构建智能体
java·spring boot·spring·springai智能体
xier_ran36 分钟前
【infra之路】02_RadixAttention与KV_Cache管理
java·spring boot·spring
黑马师兄1 小时前
RAG混合检索深度解析:让AI真正找到你要的内容
java·人工智能·ai·agent·rag·ai-native
码客日记1 小时前
Spring Boot 配置文件敏感信息加密(Jasypt 企业级完整方案)
java·spring boot·git
无限码力1 小时前
阿里算法岗 0530笔试真题 - 多约束条件下的元素匹配统计
算法·阿里笔试真题·阿里机试真题·阿里算法岗笔试