`

数据结构-排序: 选择排序(堆选择排序法)

阅读更多

数据结构-排序: 选择排序(堆选择排序法)

1.堆排序定义
 n个关键字序列Kl,K2,…,Kn称为堆,当且仅当该序列满足如下性质(简称为堆性质):
 (1) ki≤K2i且ki≤K2i+1 或(2)Ki≥K2i且ki≥K2i+1(1≤i≤ )

 若将此序列所存储的向量R[1..n]看做是一棵完全二叉树的存储结构,则堆实质上是满足如下性质的完全二叉树:树中任一非叶结点的关键字均不大于(或不小于)其左右孩子(若存在)结点的关键字。

2、大根堆和小根堆
 根结点(亦称为堆顶)的关键字是堆里所有结点关键字中最小者的堆称为小根堆。
 根结点(亦称为堆顶)的关键字是堆里所有结点关键字中最大者,称为大根堆。
注意:
 ①堆中任一子树亦是堆。
  ②以上讨论的堆实际上是二叉堆(Binary Heap),类似地可定义k叉堆。

3、堆排序特点
 堆排序(HeapSort)是一树形选择排序。
 堆排序的特点是:在排序过程中,将R[l..n]看成是一棵完全二叉树的顺序存储结构,利用完全二叉树中双亲结点和孩子结点之间的内在关系【参见二叉树的顺序存储结构】,在当前无序区中选择关键字最大(或最小)的记录。

4、堆排序与直接插入排序的区别
 直接选择排序中,为了从R[1..n]中选出关键字最小的记录,必须进行n-1次比较,然后在R[2..n]中选出关键字最小的记录,又需要做n-2次 比较。事实上,后面的n-2次比较中,有许多比较可能在前面的n-1次比较中已经做过,但由于前一趟排序时未保留这些比较结果,所以后一趟排序时又重复执 行了这些比较操作。
 堆排序可通过树形结构保存部分比较结果,可减少比较次数。

5、堆排序
堆排序利用了大根堆(或小根堆)堆顶记录的关键字最大(或最小)这一特征,使得在当前无序区中选取最大(或最小)关键字的记录变得简单。

(1)用大根堆排序的基本思想
① 先将初始文件R[1..n]建成一个大根堆,此堆为初始的无序区
② 再将关键字最大的记录R[1](即堆顶)和无序区的最后一个记录R[n]交换,由此得到新的无序区R[1..n-1]和有序区R[n],且满足R[1..n-1].keys≤R[n].key
③ 由于交换后新的根R[1]可能违反堆性质,故应将当前无序区R[1..n-1]调整为堆。然后再次将R[1..n-1]中关键字最大的记录R[1]和该区 间的最后一个记录R[n-1]交换,由此得到新的无序区R[1..n-2]和有序区R[n-1..n],且仍满足关系R[1..n- 2].keys≤R[n-1..n].keys,同样要将R[1..n-2]调整为堆。
……
直到无序区只有一个元素为止。

(2)大根堆排序算法的基本操作:
① 初始化操作:将R[1..n]构造为初始堆;
② 每一趟排序的基本操作:将当前无序区的堆顶记录R[1]和该区间的最后一个记录交换,然后将新的无序区调整为堆(亦称重建堆)。
注意:
①只需做n-1趟排序,选出较大的n-1个关键字即可以使得文件递增有序。
②用小根堆排序与利用大根堆类似,只不过其排序结果是递减有序的。堆排序和直接选择排序相反:在任何时刻,堆排序中无序区总是在有序区之前,且有序区是在原向量的尾部由后往前逐步扩大至整个向量为止。

(3)堆排序的算法:
void HeapSort(SeqIAst R)
{ //对R[1..n]进行堆排序,不妨用R[0]做暂存单元
int i;
BuildHeap(R); //将R[1-n]建成初始堆
for(i=n;i>1;i--){ //对当前无序区R[1..i]进行堆排序,共做n-1趟。
R[0]=R[1];R[1]=R[i];R[i]=R[0]; //将堆顶和堆中最后一个记录交换
  Heapify(R,1,i-1); //将R[1..i-1]重新调整为堆,仅有R[1]可能违反堆性质
} //endfor
} //HeapSort

(4) BuildHeap和Heapify函数的实现
 因为构造初始堆必须使用到调整堆的操作,先讨论Heapify的实现。
① Heapify函数思想方法
 每趟排序开始前R[l..i]是以R[1]为根的堆,在R[1]与R[i]交换后,新的无序区R[1..i-1]中只有R[1]的值发生了变化,故除 R[1]可能违反堆性质外,其余任何结点为根的子树均是堆。因此,当被调整区间是R[low..high]时,只须调整以R[low]为根的树即可。
"筛选法"调整堆
  R[low]的左、右子树(若存在)均已是堆,这两棵子树的根R[2low]和R[2low+1]分别是各自子树中关键字最大的结点。若 R[low].key不小于这两个孩子结点的关键字,则R[low]未违反堆性质,以R[low]为根的树已是堆,无须调整;否则必须将R[low]和它 的两个孩子结点中关键字较大者进行交换,即R[low]与R[large] (R[large].key=max(R[2low].key,R[2low+1].key))交换。交换后又可能使结点R[large]违反堆性质,同 样由于该结点的两棵子树(若存在)仍然是堆,故可重复上述的调整过程,对以R[large]为根的树进行调整。此过程直至当前被调整的结点已满足堆性质, 或者该结点已是叶子为止。上述过程就象过筛子一样,把较小的关键字逐层筛下去,而将较大的关键字逐层选上来。因此,有人将此方法称为"筛选法"。
  具体的算法【参见教材】

②BuildHeap的实现
  要将初始文件R[l..n]调整为一个大根堆,就必须将它所对应的完全二叉树中以每一结点为根的子树都调整为堆。
  显然只有一个结点的树是堆,而在完全二叉树中,所有序号 的结点都是叶子,因此以这些结点为根的子树均已是堆。这样,我们只需依次将以序号为 -1,…,1的结点作为根的子树都调整为堆即可。
  具体算法【参见教材】。

5、大根堆排序实例
 对于关键字序列(42,13,24,91,23,16,05,88),在建堆过程中完全二叉树及其存储结构的变化情况参见【动画演示】。

6、 算法分析
 堆排序的时间,主要由建立初始堆和反复重建堆这两部分的时间开销构成,它们均是通过调用Heapify实现的。
  堆排序的最坏时间复杂度为O(nlgn)。堆排序的平均性能较接近于最坏性能。
 由于建初始堆所需的比较次数较多,所以堆排序不适宜于记录数较少的文件。
 堆排序是就地排序,辅助空间为O(1),
 它是不稳定的排序方法。

分享到:
评论

相关推荐

    数据结构-排序算法的实现(代码+报告)

    - **定义**:堆排序是一种利用堆这种数据结构所设计的一种排序算法。 - **时间复杂度**:堆排序的平均时间复杂度为O(n log n)。 - **稳定性**:堆排序不是稳定的排序算法。 - **堆的概念**:堆是一种特殊的完全...

    数据结构经典例题:足够你做

    - 选择合适的数据结构:根据问题特点选择适合的数据结构,如用栈解决递归问题,用图处理关系网络。 - 设计算法:设计或选择合适的算法,如用贪心法解决背包问题,用深度优先搜索遍历图。 - 时间复杂度和空间...

    Java编程语言中的数据结构与算法:深入理解与实践指南.zip

    - 排序算法:包括冒泡排序、选择排序、插入排序、快速排序、归并排序和堆排序等,用于将数据按特定顺序排列。 - 搜索算法:线性搜索、二分搜索、深度优先搜索(DFS)、广度优先搜索(BFS)等,用于在数据结构中...

    数据结构.[孙凌,李丹主编

    - 堆排序:堆的概念、建堆和调整堆的操作步骤。 - 插入排序:包括简单插入排序和希尔排序两种。 #### 6. 查找算法 - 顺序查找:线性搜索的原理及其实现。 - 二分查找:适用于有序数组的查找方法。 - 散列查找:散...

    数据结构讲义

    - 定义:利用堆这种数据结构所设计的一种排序算法。 - 时间复杂度:O(n log n)。 - **归并排序** - 定义:把长度为n的输入序列分成两个长度为n/2的子序列,对这两个子序列分别采用归并排序,然后将两个排序好的子...

    数据结构--内部排序

    6. 堆排序:堆排序是一种树形选择排序,利用完全二叉树的特性构建堆,然后将堆顶元素与末尾元素交换,调整堆使其保持堆属性,再将末尾元素移除,如此循环,直到所有元素都有序。 每种排序算法都有其适用场景和优...

    用Java实现基础数据结构,排序算法、经典算法以及leetcode刷题记录_Java_下载.zip

    - 堆排序:利用堆数据结构实现,时间复杂度为O(n log n)。 3. 经典算法: 这些算法可能涵盖了搜索、图论、字符串处理等多个领域: - 深度优先搜索(DFS)和广度优先搜索(BFS):用于遍历或搜索树或图。 - 动态...

    综合排序 程序 数据结构(C语言) 课程设计

    - 堆:一种特殊的树形数据结构,满足堆属性(最大堆或最小堆),常用于优先队列的实现。 4. **排序算法的优化**: - 利用数据特性:针对有序或部分有序的数据,可以选择插入排序或冒泡排序的变种,如插入排序的二...

    数据结构知识要点

    - 堆:一种特殊的树形数据结构,满足堆属性(最大堆或最小堆)。 - 跳表:利用多级索引实现高效的查找、插入和删除操作。 7. 复杂度分析: - 时间复杂度:衡量算法执行所需时间的函数,如O(1)、O(n)、O(log n)等...

    数据结构教程-英文原版

    《数据结构教程-英文原版》是一本专为Java开发者设计的数据结构与算法教程,源自Addison Wesley出版社的第三版。这本书深入浅出地讲解了如何在Java编程环境中理解和实现各种核心数据结构和算法,是提升编程技能和...

    公共基础复习资料

    - 选择类排序法:简单选择排序、堆排序。 #### 二、程序设计基础 1. **源程序文档化**: - 注释的重要性:提高代码的可读性和维护性。 - 分类:序言性注释(文件头、模块头等)、功能性注释(代码行间解释)。 ...

    经典排序算法源代码-插入排序-选择排序-冒泡排序

    在计算机科学领域,排序算法是数据结构与算法中不可或缺的一部分,它们用于对一组数据进行排列,使其按照特定的顺序呈现。本资源包含三个经典的排序算法的源代码:插入排序、选择排序和冒泡排序,这些都是初级到中级...

    数据结构中的排序问题

    在计算机科学和信息技术领域,排序算法是数据结构中必不可少的部分,广泛应用于数据库管理、文件系统、数据分析等多个场景。 1. **排序的概念** - 排序,即对一组记录或数据元素按照某个关键字进行升序或降序的...

    数据结构 快速排序 输出每一趟结果

    根据给定文件的信息,我们可以总结出以下关于“数据结构 快速排序 输出每一趟结果”的知识点: ### 一、快速排序的基本概念 快速排序是一种高效的排序算法,采用分治法策略来把一个序列分为较小和较大的两个子序列...

    数据结构实验-排序算法

    排序算法则是数据结构中的重要部分,它们用于对一组数据进行有序排列。在这个实验中,我们将关注六种不同的排序算法:选择排序、冒泡排序、插入排序、基数排序以及快速排序和归并排序。下面是对这些排序算法的详细...

    数据结构线性表快速排序

    ### 数据结构线性表快速排序知识点解析 #### 一、数据结构基础概念 - **数据结构**:数据结构是计算机科学中的一个核心概念,它主要研究数据的逻辑结构与存储结构,以及基于这些结构的数据操作方法。良好的数据...

    排序问题(选择法排序, 冒泡法排序, 合并法排序),VB6.0源代码编写

    本主题聚焦于三种常见的排序算法:选择法排序、冒泡法排序和合并法排序,这些方法在VB6.0编程环境中都有实现的源代码。 1. **选择法排序(Selection Sort)**: - 基本思想:在未排序序列中找到最小(或最大)元素...

    山东建筑大学数据结构复习资料.zip

    - 排序:如冒泡排序、选择排序、插入排序、快速排序、归并排序、堆排序等。 - 查找:如顺序查找、二分查找、哈希查找等。 7. **文件系统**: - 逻辑文件结构:顺序文件、链接文件、索引文件等。 - 物理文件结构...

    数据结构1800例题与答案.rar

    - 数据结构:数据结构是数据的逻辑组织形式,包括线性结构(如数组、链表)、树形结构(如二叉树、堆)、图形结构等。 - 算法:解决特定问题的步骤或方法,如排序、搜索、遍历等。 2. **线性结构**: - 数组:一...

    java基础数据结构-排序算法

    // 选择排序法 public static void selectSort(Integer[] integers) { for (int i = 0; i < integers.length - 1; i++) { int minIndex = i; for (int j = i + 1; j ; j++) { if (integers[j] [minIndex]) { ...

Global site tag (gtag.js) - Google Analytics