选择排序——堆排序
一、堆排序的相关概念
1、堆的定义
从堆的定义可以看出,堆实质是满足如下性质的完全二叉树:二叉树中任一非叶子结点均小于(大于)它的孩子结点。
2、堆排序的定义
若在输出堆顶的最小值(最大值)后,使得剩余n- 1个元素的序列重又建成一个堆,则得到n个元素的次小值(次大值) ... .如此反复,便能得到一个有序序列,这个过程称之为堆排序。
二、堆的调整
1、如何在输出堆顶元素后,调整剩余元素为一个新的堆?
以小根堆为例:
①输出堆顶元素之后,以堆中最后一个元素替代之 ;
②然后将根结点值与左、右子树的根结点值进行比较,并与其中小者进行交换;
③重复上述操作②,直至叶子结点,将得到新的堆,称这个从堆顶至叶子的调整过程为"筛选"。
三、堆的建立
1、单结点的二叉树是堆;
2、在完全二叉树中所有以叶子结点(序号i > n/2)为根的子树是堆。这样,我们只需依次将以序号为n/2,n/2 - 1,......, 1的结点为根的子树均调整为堆即可。
即:对应由n个元素组成的无序序列,“筛选” 只需从第n/2个元素开始。
3、如下图所示是一个典例
4、堆的建立过程其实就是堆排序过程。
四、堆的算法描述
void HeapSort( elem R[] ) { //对R[1 ]到R[n]进行堆排序
int i;
for (i= n/2; i >=1; i--)
HeapAdjust( R,i,n); //建初始堆
for (i= n; i>1; i--){ //进行n-1趟排序
Swap(R[1], R[i]); //根与最后一 个元素交换
HeapAdjust(R,1,i-1); //对R[1]到R[i -1]重新建堆
}
} //HeapSort
void HeapAdjust (elem R[ ], int S, int m) {
/*已知R[s..m]中记录的关键字除R[s]之外均满足堆的定义,本函数调整R[s]的关键字,使R[s..m]成为一个大根堆*/
rc = R[s];
for (j=2*s;j<=m;j*= 2){ //沿key较大的孩子结点向下筛选
if(j<m&&R[j] < R[j+1]) ++j; //j为key较大的记录的下标
if( rc >= R[j] ) break;
R[s] = R[j];S =j; // rc应插入在位置s上
}//for
R[s] = rc; //插入
} // HeapAdjust