在计算机科学与数学中,排序算法是一种基本并且常用的算法,一个排序演算法是一种能将一串资料依照特定排序方式的一种演算法。有效的排序演算法在一些演算法中是重要的,如此这些演算法才能得到正确解答。排序演算法也用在处理文字资料以及产生人类可读的输出结果。由于实际工作中处理的数量巨大,所以排序算法对算法本身的速度要求很高。现介绍C/C++中几种常见排序算法的简单运用方法。
插入排序是最简单最直观的排序算法了,它的依据是:遍历到第N个元素的时候前面的N-1个元素已经是排序好的了,那么就查找前面的N-1个元素把这第N个元素放在合适的位置,如此下去直到遍历完序列的元素为止。
算法的复杂度也是简单的,排序第一个需要1的复杂度,排序第二个需要2的复杂度,因此整个的复杂度就是
1 + 2 + 3 + …… + N = O(N ^ 2)的复杂度。
// 插入排序 for (i = 1; i < length; i++) |
shell排序是对插入排序的一个改装,它每次排序把序列的元素按照某个增量分成几个子序列,对这几个子序列进行插入排序,然后不断的缩小增量扩大每个子序列的元素数量,直到增量为一的时候子序列就和原先的待排列序列一样了,此时只需要做少量的比较和移动就可以完成对序列的排序了。
// shell排序 // 增量从数组长度的一半开始,每次减小一倍 |
动画演示:
http://202.113.89.254/DataStructure/DS/web/flashhtml/shell.htm
堆的定义:
n个关键字序列Kl,K2,…,Kn称为堆,当且仅当该序列满足如下性质(简称为堆性质):
(1) ki≤K2i且ki≤K2i+1 或(2)Ki≥K2i且ki≥K2i+1(1≤i≤)
若将此序列所存储的向量R[1……n]看做是一棵完全二叉树的存储结构,则堆实质上是满足如下性质的完全二叉树:树中任一非叶结点的关键字均不大于(或不小于)其左右孩子(若存在)结点的关键字。
堆的这个性质使得可以迅速定位在一个序列之中的最小(大)的元素。
堆排序算法的过程如下:1)得到当前序列的最小(大)的元素 2)把这个元素和最后一个元素进行交换,这样当前的最小(大)的元素就放在了序列的最后,而原先的最后一个元素放到了序列的最前面 3)的交换可能会破坏堆序列的性质(注意此时的序列是除去已经放在最后面的元素),因此需要对序列进行调整,使之满足于上面堆的性质。重复上面的过程,直到序列调整完毕为止。
// array是待调整的堆数组,i是待调整的数组元素的位置,length是数组的长度 for (nTemp = array[i]; 2 * i + 1 < nLength; i = nChild) // 得到子结点中较大的结点 // 如果较大的子结点大于父结点那么把较大的子结点往上移动,替换它的父结点 // 最后把需要调整的元素值放到合适的位置 // 堆排序算法 // 从最后一个元素开始对序列进行调整,不断的缩小调整的范围直到第一个元素 // 不断缩小调整heap的范围,每一次调整完毕保证第一个元素是当前序列的最大值 |
一个测试及输出的结果,在每次HeapAdjust之后显示出来当前数组的情况
before Heap sort: 71 18 151 138 160 63 174 169 79 78 // 开始调整前半段数组元素 71 18 151 138 160 63 174 169 79 78 71 18 151 169 160 63 174 138 79 78 71 18 174 169 160 63 151 138 79 78 71 169 174 138 160 63 151 18 79 78 174 169 151 138 160 63 71 18 79 78 // 开始进行全局的调整 169 160 151 138 78 63 71 18 79 174 160 138 151 79 78 63 71 18 169 174 151 138 71 79 78 63 18 160 169 174 138 79 71 18 78 63 151 160 169 174 79 78 71 18 63 138 151 160 169 174 78 63 71 18 79 138 151 160 169 174 71 63 18 78 79 138 151 160 169 174 63 18 71 78 79 138 151 160 169 174 18 63 71 78 79 138 151 160 169 174 |
动画演示:
http://202.113.89.254/DataStructure/DS/web/flashhtml/duipaixu.htm
冒泡排序算法的思想:很简单,每次遍历完序列都把最大(小)的元素放在最前面,然后再对剩下的序列从父前面的一个过程,每次遍历完之后待排序序列就少一个元素,当待排序序列减小为只有一个元素的时候排序就结束了。因此,复杂度在最坏的情况下是O(N ^ 2)。
void Swap( int * a, int * b) temp = * a; |
快速排序的算法思想: 选定一个枢纽元素,对待排序序列进行分割,分割之后的序列一个部分小于枢纽元素,一个部分大于枢纽元素,再对这两个分割好的子序列进行上述的过程。
// 对一个给定范围的子序列选定一个枢纽元素,执行完函数之后返回分割元素所在的位置, while (low < high) // 将这个比枢纽元素小的元素交换到前半部分 // 从前往后在前半部分中寻找第一个大于枢纽元素的元素 // 将这个比枢纽元素大的元素交换到后半部分 // 返回枢纽元素所在的位置 // 快速排序 |
归并排序的算法思想:把待排序序列分成相同大小的两个部分,依次对这两部分进行归并排序,完毕之后再按照顺序进行合并。
// 归并排序中的合并算法 // 拷贝前半部分数组 // 归并排序 |