当前位置:  编程技术>c/c++/嵌入式

堆排序算法(选择排序改进)

    来源: 互联网  发布时间:2014-10-23

    本文导语:  首先要理解堆的含义:要么所有节点都不大于其子孩子节点数据,要么都不小于其子孩子节点数据 堆排序的核心思想:就是要满足所有节点都满足上面两点,如何完成,看下面 堆排序的步骤: 1.首先要建成一个大顶堆或者小...

首先要理解堆的含义:要么所有节点都不大于其子孩子节点数据,要么都不小于其子孩子节点数据

堆排序的核心思想:就是要满足所有节点都满足上面两点,如何完成,看下面

堆排序的步骤:

1.首先要建成一个大顶堆或者小顶堆,在建的过程中其实就是调整节点的位置,首先要从最后最后一个节点的母亲节点开始,按照堆的含义调整。为什么不是最后一个或者其他?因为要保证完整性和不必要性,所以只需从最后一个的母亲节点开始即可(下面的堆默认存在顺序结构,从索引0开始的,所以有些二叉树的特性请查阅二叉树),直至索引节点为0的节点。调整完成后即成为一个堆,但是这里的数据并没有排序好,所以下一部调整顺序。

2.从最后一个数据开始,与第一个数据进行交换,然后按照堆的含义调整第一个数据。为什么先选择最后一个数据?因为默认情况下,最后一个或者是较大或者是较小,可以满足调整要求。这时就考虑当前所有数据减去最后一个,因为这个已是最大或者是最小,不必再考虑.。直至调整没有任何数据,此时已完成排序。

具体图例不再标识,有此爱好可以参考其他书籍或者网上的介绍,下面看堆排序代码:

代码如下:

int HeapSort(MergeType* L)
{
 int i = 0;
 if (!L->elem)
 {
  return -1;
 }

 //创建堆
 for (int i = L->len/2-1; i >= 0; i--)
 {
  HeapAdjust(L, i, L->len-1);
 }

 //堆排序
 for (i = L->len-1; i >= 0; i-- )
 {
  swap(L->elem[i], L->elem[0]);
  HeapAdjust(L, 0, i-1);
 }
 return 0; 
}

注意:
1)由于父子节点的关系,for循环第一个数据索引其实是L,len-1,但是其父母节点(i)与 当前节点(p)的关系:p = 2i+1 或者2i+2; 如果存储数据的节点第一个索引不是0而是1,这里p=2i或者p=2i+1,请参看有关书籍的证明,所以当前父母节点:i =(p-1)/ 2 = (L.len-1-1)/2 = L.len/2-1

2)由于再次调整数据的时候是从最后一个数据,所以需要交换数据swap,再进行当前顶点数据也就是第一个数据的堆调整,但是此时调整的对象只是(0~i)这些数据,其他已经排序好,所以不再需要调整

下面看一下调整代码,如下:

代码如下:

int HeapAdjust(MergeType* L, int nPos, int nEnd)
{
 for (int i = nPos*2+1; i < nEnd ; i = 2*i+1)
 {
  if (L->elem[i] elem[i+1])
  {
   i++;
  }
  if (L->elem[nPos] >= L->elem[i])
  {
   break;
  }
  swap(L->elem[nPos], L->elem[i]);
  nPos = i;
 }
 return 0;
}

这里使用的是在一个层次上是数据直接交换,其实这不是必须的,因为最后才把数据放到最后的位置,所以也可以使用下面的代码,减少复制的次数

代码如下:

int HeapAdjustEx(MergeType* L, int nPos, int nEnd)
{
 int nTempkey = L->elem[nPos];

 for (int i = nPos*2+1; i < nEnd ; i = 2*i+1)
 {
  if (L->elem[i] elem[i+1])//选出最大的子孩子
  {
   i++;
  }
  if (nTempkey >= L->elem[i]) //如果当前节点大于最大子孩子退出
  {
   break;
  }
  L->elem[nPos] = L->elem[i]; //否则进行数据交换
  nPos = i;
 }
 L->elem[nPos] = nTempkey;
 return 0;
}

这里就可以减少较多的复制操作,也就是俗称的移动操作次数;这里for循环的起始节点按照上面的推论,子节点应该为p=2i+1,所以第一个应该为2*nPos+1,对应当前要比较节点的做孩子,右孩子为2*nPos+2,也就是左孩子+1,其他请看注释。
时间复杂度:O(nlogn),分析过程暂略


    
 
 

您可能感兴趣的文章:

  • <<大话数据结构>>中冒泡排序算法改进
  • java 合并排序算法、冒泡排序算法、选择排序算法、插入排序算法、快速排序算法的描述
  • python算法学习之桶排序算法实例(分块排序)
  • 可视化算法排序过程 Sound of Sorting
  • 常用排序算法整理分享(快速排序算法、希尔排序)
  • C#排序算法之快速排序
  • C++实现顺序排序算法简单示例代码
  • 排序算法之PHP版快速排序、冒泡排序
  • 算法之排序算法的算法思想和使用场景总结
  • VC++实现选择排序算法简单示例
  • php冒泡排序算法实现代码
  • php排序算法 PHP版快速排序与冒泡排序
  • 希尔排序的算法代码
  • python 算法 排序实现快速排序
  • php选择排序算法实现代码
  • python 实现插入排序算法
  • python实现排序算法
  • python冒泡排序算法的实现代码
  • python插入排序算法的实现代码
  • python计数排序和基数排序算法实例
  • C++选择排序算法实例
  • C++ 冒泡排序数据结构、算法及改进算法
  •  
    本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
    本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • C++ Lists(链表) 成员 sort():给list排序
  • Java中的数组排序方式(快速排序、冒泡排序、选择排序)
  • STL vector+sort排序和multiset/multimap排序比较
  • java数组排序示例(冒泡排序、快速排序、希尔排序、选择排序)
  • java map(HashMap TreeMap)用法:初始化,遍历和排序详解
  • 问题:DefaulTableModel是否有排序的功能,如果没有,jTable如何排序,我是从XML取数据到Table里。
  • linux下top命令详解包括top命令参数使用及结果(virt,res,shr)排序举例说明
  • PHP快速排序小例子 php快速排序实现方法
  • 数据库查询排序使用随机排序结果示例(Oracle/MySQL/MS SQL Server)
  • 深入Java冒泡排序与选择排序的区别详解
  • C#中使用快速排序按文件创建时间将文件排序的源码
  • 用c语言实现冒泡排序,选择排序,快速排序
  • php数组随机排序示例
  • jQuery表格排序插件 tablesorter
  • jQuery排序工具 jQuery.sorted
  • 关于awk数组的排序
  • Eclipse文本排序插件 SortIt
  • linux中对文件排序的命令(文件夹中包含子文件)
  • 请问怎么用sort对多个字段进行排序?
  • ll 命令输出,使用sort排序问题
  • jQuery 表格排序插件 Stupid Table


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3