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

所属分类: 软件编程 / C 语言 阅读数: 113
收藏 0 赞 0 分享

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

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

堆排序的步骤:

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] <= L->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] <= L->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),分析过程暂略

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多