zoukankan      html  css  js  c++  java
  • 黑马程序猿-----Java之你不得不知道的排序

    ------<a href="http://www.itheima.com" target="blank">Java培训、Android培训、iOS培训、.Net培训</a>、期待与您交流!

    -------

     

     

    排序有内部排序和外部排序。内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据非常大,一次不能容纳所有的排序记录,在排序过程中须要訪问外存。


    我们这里说说八大排序就是内部排序。




    当n较大,则应採用时间复杂度为O(nlog2n)的排序方法:高速排序、堆排序或归并排序序。


    高速排序:是眼下基于比較的内部排序中被觉得是最好的方法,当待排序的keyword是随机分布时,高速排序的平均时间最短。


    1.插入排序—直接插入排序(Straight Insertion Sort)


    基本思想:


    将一个记录插入到已排序好的有序表中,从而得到一个新,记录数增1的有序表。

    即:先将序列的第1个记录看成是一个有序的子序列。然后从第2个记录逐个进行插入,直至整个序列有序为止。


    要点:设立哨兵,作为暂时存储和推断数组边界之用。


    直接插入排序演示样例:




    假设碰见一个和插入元素相等的,那么插入元素把想插入的元素放在相等元素的后面。所以。相等元素的前后顺序没有改变,从原无序序列出去的顺序就是排好序后的顺序,所以插入排序是稳定的。


    算法的实现:

    .NET、void print(int a[], int n ,int i){

    cout<<i <<":";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }



    void InsertSort(int a[], int n)

    {

    for(int i= 1; i<n; i++){

    if(a[i] < a[i-1]){ //若第i个元素大于i-1元素。直接插入。

    小于的话。移动有序表后插入

    int j= i-1;

    int x = a[i]; //复制为哨兵。即存储待排序元素

    a[i] = a[i-1]; //先后移一个元素

    while(x < a[j]){ //查找在有序表的插入位置

    a[j+1] = a[j];

    j--; //元素后移

    }

    a[j+1] = x; //插入到正确位置

    }

    print(a,n,i); //打印每趟排序的结果

    }


    }


    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    InsertSort(a,8);

    print(a,8,8);

    }

    效率:


    时间复杂度:O(n^2).


    其它的插入排序有二分插入排序,2-路插入排序。


    2.插入排序—希尔排序(Shell`s Sort)


    希尔排序是1959 年由D.L.Shell 提出来的,相对直接排序有较大的改进。

    希尔排序又叫缩小增量排序


    基本思想:


    先将整个待排序的记录序列切割成为若干子序列分别进行直接插入排序。待整个序列中的记录“基本有序”时,再对全体记录进行依次直接插入排序。


    操作方法:


    1.选择一个增量序列t1,t2,…。tk,当中ti>tj,tk=1;

    2.按增量序列个数k,对序列进行k 趟排序;

    3.每趟排序。依据相应的增量ti,将待排序列切割成若干长度为m 的子序列。分别对各子表进行直接插入排序。

    仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。


    希尔排序的演示样例:




    算法实现:


    我们简单处理增量序列:增量序列d = {n/2 ,n/4, n/8 .....1} n为要排序数的个数


    即:先将要排序的一组记录按某个增量d(n/2,n为要排序数的个数)分成若干组子序列。每组中记录的下标相差d.对每组中所有元素进行直接插入排序,然后再用一个较小的增量(d/2)对它进行分组,在每组中再进行直接插入排序。

    继续不断缩小增量直至为1,最后使用直接插入排序完毕排序。

    void print(int a[], int n ,int i){

    cout<<i <<":";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }

    /**

    * 直接插入排序的一般形式

    *

    * @param int dk 缩小增量,假设是直接插入排序,dk=1

    *

    */


    void ShellInsertSort(int a[], int n, int dk)

    {

    for(int i= dk; i<n; ++i){

    if(a[i] < a[i-dk]){ //若第i个元素大于i-1元素。直接插入。

    小于的话,移动有序表后插入

    int j = i-dk;

    int x = a[i]; //复制为哨兵,即存储待排序元素

    a[i] = a[i-dk]; //首先后移一个元素

    while(x < a[j]){ //查找在有序表的插入位置

    a[j+dk] = a[j];

    j -= dk; //元素后移

    }

    a[j+dk] = x; //插入到正确位置

    }

    print(a, n,i );

    }


    }


    /**

    * 先按增量d(n/2,n为要排序数的个数进行希尔排序

    *

    */

    void shellSort(int a[], int n){


    int dk = n/2;

    while( dk >= 1 ){

    ShellInsertSort(a, n, dk);

    dk = dk/2;

    }

    }

    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    //ShellInsertSort(a,8,1); //直接插入排序

    shellSort(a,8); //希尔插入排序

    print(a,8,8);

    }

    希尔排序时效分析非常难。关键码的比較次数与记录移动次数依赖于增量因子序列d的选取,特定情况下能够准确估算出关键码的比較次数和记录的移动次数。

    眼下还没有人给出选取最好的增量因子序列的方法。

    增量因子序列能够有各种取法,有取奇数的,也有取质数的,但须要注意:增量因子中除1 外没有公因子,且最后一个增量因子必须为1。希尔排序方法是一个不稳定的排序方法。


    3.选择排序—简单选择排序(Simple Selection Sort)


    基本思想:


    在要排序的一组数中。选出最小(或者最大)的一个数与第1个位置的数交换;然后在剩下的数其中再找最小(或者最大)的与第2个位置的数交换,依次类推,直到第n-1个元素(倒数第二个数)和第n个元素(最后一个数)比較为止。


    简单选择排序的演示样例:



    操作方法:


    第一趟,从n 个记录中找出关键码最小的记录与第一个记录交换;

    第二趟,从第二个记录開始的n-1 个记录中再选出关键码最小的记录与第二个记录交换。


    以此类推.....

    第i 趟,则从第i 个记录開始的n-i+1 个记录中选出关键码最小的记录与第i 个记录交换,


    直到整个序列按关键码有序。


    算法实现:

    void print(int a[], int n ,int i){

    cout<<"第"<<i+1 <<"趟 : ";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }

    /**

    * 数组的最小值

    *

    * @return int 数组的键值

    */

    int SelectMinKey(int a[], int n, int i)

    {

    int k = i;

    for(int j=i+1 ;j< n; ++j) {

    if(a[k] > a[j]) k = j;

    }

    return k;

    }


    /**

    * 选择排序

    *

    */

    void selectSort(int a[], int n){

    int key, tmp;

    for(int i = 0; i< n; ++i) {

    key = SelectMinKey(a, n,i); //选择最小的元素

    if(key != i){

    tmp = a[i]; a[i] = a[key]; a[key] = tmp; //最小元素与第i位置元素互换

    }

    print(a, n , i);

    }

    }

    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    cout<<"初始值:";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl<<endl;

    selectSort(a, 8);

    print(a,8,8);

    }

    简单选择排序的改进——二元选择排序


    简单选择排序,每趟循环仅仅能确定一个元素排序后的定位。

    我们能够考虑改进为每趟循环确定两个元素(当前趟最大和最小记录)的位置,从而降低排序所需的循环次数。

    改进后对n个数据进行排序。最多仅仅需进行[n/2]趟循环就可以。

    详细实现例如以下:

    void SelectSort(int r[],int n) {

    int i ,j , min ,max, tmp;

    for (i=1 ;i <= n/2;i++) {

    // 做不超过n/2趟选择排序

    min = i; max = i ; //分别记录最大和最小keyword记录位置

    for (j= i+1; j<= n-i; j++) {

    if (r[j] > r[max]) {

    max = j ; continue ;

    }

    if (r[j]< r[min]) {

    min = j ;

    }

    }

    //该交换操作还可分情况讨论以提高效率

    tmp = r[i-1]; r[i-1] = r[min]; r[min] = tmp;

    tmp = r[n-i]; r[n-i] = r[max]; r[max] = tmp;


    }

    }

    4.选择排序—堆排序(Heap Sort)


    堆排序是一种树形选择排序,是对直接选择排序的有效改进。


    基本思想:


    堆的定义例如以下:具有n个元素的序列(k1,k2,...,kn),当且仅当满足




    时称之为堆。

    由堆的定义能够看出,堆顶元素(即第一个元素)必为最小项(小顶堆)。


    若以一维数组存储一个堆。则堆相应一棵全然二叉树。且全部非叶结点的值均不大于(或不小于)其子女的值,根结点(堆顶元素)的值是最小(或最大)的。如:


    (a)大顶堆序列:(96, 83,27,38,11,09)

    (b) 小顶堆序列:(12。36。24,85。47。30,53,91)




    初始时把要排序的n个数的序列看作是一棵顺序存储的二叉树(一维数组存储二叉树),调整它们的存储序,使之成为一个堆,将堆顶元素输出,得到n 个元素中最小(或最大)的元素,这时堆的根节点的数最小(或者最大)。

    然后对前面(n-1)个元素又一次调整使之成为堆,输出堆顶元素。得到n 个元素中次小(或次大)的元素。

    依此类推,直到仅仅有两个节点的堆,并对它们作交换。最后得到有n个节点的有序序列。

    称这个过程为堆排序。


    因此,实现堆排序需解决两个问题:


    1. 怎样将n 个待排序的数建成堆;
    2. 输出堆顶元素后。如何调整剩余n-1 个元素,使其成为一个新堆。


    首先讨论第二个问题:输出堆顶元素后。对剩余n-1元素又一次建成堆的调整过程。


    调整小顶堆的方法:


    1)设有m 个元素的堆,输出堆顶元素后。剩下m-1 个元素。

    将堆底元素送入堆顶((最后一个元素与堆顶进行交换),堆被破坏,其原因仅是根结点不满足堆的性质。


    2)将根结点与左、右子树中较小元素的进行交换。


    3)若与左子树交换:假设左子树堆被破坏,即左子树的根结点不满足堆的性质。则反复方法 (2).


    4)若与右子树交换,假设右子树堆被破坏。即右子树的根结点不满足堆的性质。则反复方法 (2).


    5)继续对不满足堆性质的子树进行上述交换操作,直到叶子结点。堆被建成。


    称这个自根结点到叶子结点的调整过程为筛选。如图:




    再讨论对n 个元素初始建堆的过程。


    建堆方法:对初始序列建堆的过程,就是一个重复进行筛选的过程。


    1)n 个结点的全然二叉树,则最后一个结点是第个结点的子树。

    2)筛选从第个结点为根的子树開始。该子树成为堆。

    3)之后向前依次对各结点为根的子树进行筛选。使之成为堆,直到根结点。


    如图建堆初始过程:无序序列:(49,38。65,97,76,13。27,49)





    算法的实现:


    从算法描写叙述来看,堆排序须要两个过程,一是建立堆,二是堆顶与堆的最后一个元素交换位置。所以堆排序有两个函数组成。

    一是建堆的渗透函数,二是重复调用渗透函数实现排序的函数。

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }




    /**

    * 已知H[s…m]除了H[s] 外均满足堆的定义

    * 调整H[s],使其成为大顶堆.即将对第s个结点为根的子树筛选,

    *

    * @param H是待调整的堆数组

    * @param s是待调整的数组元素的位置

    * @param length是数组的长度

    *

    */

    void HeapAdjust(int H[],int s, int length)

    {

    int tmp = H[s];

    int child = 2*s+1; //左孩子结点的位置。

    (i+1 为当前调整结点的右孩子结点的位置)

    while (child < length) {

    if(child+1 <length && H[child]<H[child+1]) { // 假设右孩子大于左孩子(找到比当前待调整结点大的孩子结点)

    ++child ;

    }

    if(H[s]<H[child]) { // 假设较大的子结点大于父结点

    H[s] = H[child]; // 那么把较大的子结点往上移动,替换它的父结点

    s = child; // 又一次设置s ,即待调整的下一个结点的位置

    child = 2*s+1;

    } else { // 假设当前待调整结点大于它的左右孩子。则不须要调整,直接退出

    break;

    }

    H[s] = tmp; // 当前待调整的结点放到比其大的孩子结点位置上

    }

    print(H,length);

    }



    /**

    * 初始堆进行调整

    * 将H[0..length-1]建成堆

    * 调整完之后第一个元素是序列的最小的元素

    */

    void BuildingHeap(int H[], int length)

    {

    //最后一个有孩子的节点的位置 i= (length -1) / 2

    for (int i = (length -1) / 2 ; i >= 0; --i)

    HeapAdjust(H,i,length);

    }

    /**

    * 堆排序算法

    */

    void HeapSort(int H[],int length)

    {

    //初始堆

    BuildingHeap(H, length);

    //从最后一个元素開始对序列进行调整

    for (int i = length - 1; i > 0; --i)

    {

    //交换堆顶元素H[0]和堆中最后一个元素

    int temp = H[i]; H[i] = H[0]; H[0] = temp;

    //每次交换堆顶元素和堆中最后一个元素之后,都要对堆进行调整

    HeapAdjust(H,0,i);

    }

    }


    int main(){

    int H[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(H,10);

    HeapSort(H,10);

    //selectSort(a, 8);

    cout<<"结果:";

    print(H,10);


    }

    分析:


    设树深度为k,。从根到叶的筛选。元素比較次数至多2(k-1)次,交换记录至多k 次。

    所以,在建好堆后,排序过程中的筛选次数不超过下式:



    而建堆时的比較次数不超过4n 次。因此堆排序最坏情况下,时间复杂度也为:O(nlogn )。


    5.交换排序—冒泡排序(Bubble Sort)


    基本思想:


    在要排序的一组数中。对当前还未排好序的范围内的所有数。自上而下对相邻的两个数依次进行比較和调整,让较大的数往下沉,较小的往上冒。

    即:每当两相邻的数比較后发现它们的排序与排序要求相反时,就将它们互换。


    冒泡排序的演示样例:




    算法的实现:

    void bubbleSort(int a[], int n){

    for(int i =0 ; i< n-1; ++i) {

    for(int j = 0; j < n-i-1; ++j) {

    if(a[j] > a[j+1])

    {

    int tmp = a[j] ; a[j] = a[j+1] ; a[j+1] = tmp;

    }

    }

    }

    }

    冒泡排序算法的改进


    对冒泡排序常见的改进方法是增加一标志性变量exchange,用于标志某一趟排序过程中是否有数据交换,假设进行某一趟排序时并没有进行数据交换。则说明数据已经按要求排列好,可马上结束排序,避免不必要的比較过程。本文再提供下面两种改进算法:


    1.设置一标志性变量pos,用于记录每趟排序中最后一次进行交换的位置。因为pos位置之后的记录均已交换到位,故在进行下一趟排序时仅仅要扫描到pos位置就可以。


    改进后算法例如以下:

    void Bubble_1 ( int r[], int n) {

    int i= n -1; //初始时,最后位置保持不变

    while ( i> 0) {

    int pos= 0; //每趟開始时,无记录交换

    for (int j= 0; j< i; j++)

    if (r[j]> r[j+1]) {

    pos= j; //记录交换的位置

    int tmp = r[j]; r[j]=r[j+1];r[j+1]=tmp;

    }

    i= pos; //为下一趟排序作准备

    }

    }

    2.传统冒泡排序中每一趟排序操作仅仅能找到一个最大值或最小值,我们考虑利用在每趟排序中进行正向和反向两遍冒泡的方法一次能够得到两个终于值(最大者和最小者) , 从而使排序趟数差点儿降低了一半。


    改进后的算法实现为:

    void Bubble_2 ( int r[], int n){

    int low = 0;

    int high= n -1; //设置变量的初始值

    int tmp,j;

    while (low < high) {

    for (j= low; j< high; ++j) //正向冒泡,找到最大者

    if (r[j]> r[j+1]) {

    tmp = r[j]; r[j]=r[j+1];r[j+1]=tmp;

    }

    --high; //改动high值, 前移一位

    for ( j=high; j>low; --j) //反向冒泡,找到最小者

    if (r[j]<r[j-1]) {

    tmp = r[j]; r[j]=r[j-1];r[j-1]=tmp;

    }

    ++low; //改动low值,后移一位

    }

    }

    6.交换排序—高速排序(Quick Sort)


    基本思想:


    1)选择一个基准元素,通常选择第一个元素或者最后一个元素,

    2)通过一趟排序讲待排序的记录切割成独立的两部分。当中一部分记录的元素值均比基准元素值小。

    还有一部分记录的 元素值比基准值大。

    3)此时基准元素在其排好序后的正确位置

    4)然后分别对这两部分记录用相同的方法继续进行排序。直到整个序列有序。


    高速排序的演示样例:


    (a)一趟排序的过程:




    (b)排序的全过程




    算法的实现:


    递归实现:

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    void swap(int *a, int *b)

    {

    int tmp = *a;

    *a = *b;

    *b = tmp;

    }


    int partition(int a[], int low, int high)

    {

    int privotKey = a[low]; //基准元素

    while(low < high){ //从表的两端交替地向中间扫描

    while(low < high && a[high] >= privotKey) --high; //从high 所指位置向前搜索,至多到low+1 位置。

    将比基准元素小的交换到低端

    swap(&a[low], &a[high]);

    while(low < high && a[low] <= privotKey ) ++low;

    swap(&a[low], &a[high]);

    }

    print(a,10);

    return low;

    }



    void quickSort(int a[], int low, int high){

    if(low < high){

    int privotLoc = partition(a, low, high); //将表一分为二

    quickSort(a, low, privotLoc -1); //递归对低子表递归排序

    quickSort(a, privotLoc + 1, high); //递归对高子表递归排序

    }

    }


    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(a,10);

    quickSort(a,0,9);

    cout<<"结果:";

    print(a,10);


    }

    分析:


    高速排序是通常被觉得在同数量级(O(nlog2n))的排序方法中平均性能最好的。但若初始序列按关键码有序或基本有序时,快排序反而蜕化为冒泡排序。为改进之。通常以“三者取中法”来选取基准记录。即将排序区间的两个端点与中点三个记录关键码居中的调整为支点记录。高速排序是一个不稳定的排序方法。


    高速排序的改进


    在本改进算法中,仅仅对长度大于k的子序列递归调用高速排序,让原序列基本有序,然后再对整个基本有序序列用插入排序算法排序。

    实践证明。改进后的算法时间复杂度有所减少。且当k取值为 8 左右时,改进算法的性能最佳。算法思想例如以下:

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    void swap(int *a, int *b)

    {

    int tmp = *a;

    *a = *b;

    *b = tmp;

    }


    int partition(int a[], int low, int high)

    {

    int privotKey = a[low]; //基准元素

    while(low < high){ //从表的两端交替地向中间扫描

    while(low < high && a[high] >= privotKey) --high; //从high 所指位置向前搜索,至多到low+1 位置。将比基准元素小的交换到低端

    swap(&a[low], &a[high]);

    while(low < high && a[low] <= privotKey ) ++low;

    swap(&a[low], &a[high]);

    }

    print(a,10);

    return low;

    }



    void qsort_improve(int r[ ],int low,int high, int k){

    if( high -low > k ) { //长度大于k时递归, k为指定的数

    int pivot = partition(r, low, high); // 调用的Partition算法保持不变

    qsort_improve(r, low, pivot - 1,k);

    qsort_improve(r, pivot + 1, high,k);

    }

    }

    void quickSort(int r[], int n, int k){

    qsort_improve(r,0,n,k);//先调用改进算法Qsort使之基本有序


    //再用插入排序对基本有序序列排序

    for(int i=1; i<=n;i ++){

    int tmp = r[i];

    int j=i-1;

    while(tmp < r[j]){

    r[j+1]=r[j]; j=j-1;

    }

    r[j+1] = tmp;

    }


    }




    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(a,10);

    quickSort(a,9,4);

    cout<<"结果:";

    print(a,10);


    }

    7.归并排序(Merge Sort)


    基本思想:


    归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表。即把待排序序列分为若干个子序列。每一个子序列是有序的。

    然后再把有序子序列合并为总体有序序列。


    归并排序演示样例:




    合并方法:


    设r[i…n]由两个有序子表r[i…m]和r[m+1…n]组成,两个子表长度分别为n-i +1、n-m。


    1.j=m+1;k=i。i=i; //置两个子表的起始下标及辅助数组的起始下标


    2.若i>m 或j>n。转⑷ //当中一个子表已合并完,比較选取结束


    3.//选取r[i]和r[j]较小的存入辅助数组rf

    假设r[i]<r[j]。rf[k]=r[i]; i++; k++。 转⑵

    否则,rf[k]=r[j]; j++; k++。 转⑵


    4.//将尚未处理完的子表中元素存入rf

    假设i<=m,将r[i…m]存入rf[k…n] //前一子表非空
    假设j<=n , 将r[j…n] 存入rf[k…n] //后一子表非空


    5.合并结束。

    //将r[i…m]和r[m +1 …n]归并到辅助数组rf[i…n]

    void Merge(ElemType *r,ElemType *rf, int i, int m, int n)

    {

    int j,k;

    for(j=m+1,k=i; i<=m && j <=n ; ++k){

    if(r[j] < r[i]) rf[k] = r[j++];

    else rf[k] = r[i++];

    }

    while(i <= m) rf[k++] = r[i++];

    while(j <= n) rf[k++] = r[j++];

    }

    归并的迭代算法


    1个元素的表总是有序的。

    所以对n 个元素的待排序列。每一个元素可看成1 个有序子表。

    对子表两两合并生成n/2个子表,所得子表除最后一个子表长度可能为1 外,其余子表长度均为2。再进行两两合并,直到生成n 个元素按关键码有序的表。

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    //将r[i…m]和r[m +1 …n]归并到辅助数组rf[i…n]

    void Merge(ElemType *r,ElemType *rf, int i, int m, int n)

    {

    int j,k;

    for(j=m+1,k=i; i<=m && j <=n ; ++k){

    if(r[j] < r[i]) rf[k] = r[j++];

    else rf[k] = r[i++];

    }

    while(i <= m) rf[k++] = r[i++];

    while(j <= n) rf[k++] = r[j++];

    print(rf,n+1);

    }


    void MergeSort(ElemType *r, ElemType *rf, int lenght)

    {

    int len = 1;

    ElemType *q = r ;

    ElemType *tmp ;

    while(len < lenght) {

    int s = len;

    len = 2 * s ;

    int i = 0;

    while(i+ len <lenght){

    Merge(q, rf, i, i+ s-1, i+ len-1 ); //对等长的两个子表合并

    i = i+ len;

    }

    if(i + s < lenght){

    Merge(q, rf, i, i+ s -1, lenght -1); //对不等长的两个子表合并

    }

    tmp = q; q = rf; rf = tmp; //交换q,rf。以保证下一趟归并时,仍从q 归并到rf

    }

    }



    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    int b[10];

    MergeSort(a, b, 10);

    print(b,10);

    cout<<"结果:";

    print(a,10);


    }

    两路归并的递归算法

    void MSort(ElemType *r, ElemType *rf,int s, int t)

    {

    ElemType *rf2;

    if(s==t) r[s] = rf[s];

    else

    {

    int m=(s+t)/2; /*平分*p 表*/

    MSort(r, rf2, s, m); /*递归地将p[s…m]归并为有序的p2[s…m]*/

    MSort(r, rf2, m+1, t); /*递归地将p[m+1…t]归并为有序的p2[m+1…t]*/

    Merge(rf2, rf, s, m+1,t); /*将p2[s…m]和p2[m+1…t]归并到p1[s…t]*/

    }

    }

    void MergeSort_recursive(ElemType *r, ElemType *rf, int n)

    { /*对顺序表*p 作归并排序*/

    MSort(r, rf,0, n-1);

    }

    8.桶排序/基数排序(Radix Sort)


    说基数排序之前,我们先说桶排序:


    基本思想:是将阵列分到有限数量的桶子里。每一个桶子再个别排序(有可能再使用别的排序算法或是以递回方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。

    当要被排序的阵列内的数值是均匀分配的时候,桶排序使用线性时间(Θ(n))。但桶排序并非 比較排序,他不受到 O(n log n) 下限的影响。


    简单来说。就是把数据分组,放在一个个的桶中,然后对每一个桶里面的在进行排序。


    比如要对大小为[1..1000]范围内的n个整数A[1..n]排序


    首先,能够把桶设为大小为10的范围。详细而言,设集合B[1]存储[1..10]的整数,集合B[2]存储 (10..20]的整数,……集合B[i]存储( (i-1)*10, i*10]的整数。i = 1,2,..100。总共同拥有 100个桶。


    然后,对A[1..n]从头到尾扫描一遍,把每一个A[i]放入相应的桶B[j]中。 再对这100个桶中每一个桶里的数字排序,这时可用冒泡,选择。乃至快排,一般来说任 何排序法都能够。


    最后,依次输出每一个桶里面的数字,且每一个桶中的数字从小到大输出。这 样就得到全部数字排好序的一个序列了。


    假设有n个数字,有m个桶,假设数字是平均分布的,则每一个桶里面平均有n/m个数字。

    假设对每一个桶中的数字採用高速排序,那么整个算法的复杂度是 O(n + m * n/m*log(n/m)) = O(n + nlogn - nlogm) 从上式看出,当m接近n的时候,桶排序复杂度接近O(n)


    当然,以上复杂度的计算是基于输入的n个数字是平均分布这个如果的。

    这个如果是非常强的 。实际应用中效果并没有这么好。如果全部的数字都落在同一个桶中,那就退化成一般的排序了。


    前面说的几大排序算法 ,大部分时间复杂度都是O(n2)。也有部分排序算法时间复杂度是O(nlogn)。而桶式排序却能实现O(n)的时间复杂度。但桶排序的缺点是:


    1)首先是空间复杂度比較高。须要的额外开销大。排序有两个数组的空间开销,一个存放待排序数组。一个就是所谓的桶,比方待排序值是从0到m-1,那就须要m个桶。这个桶数组就要至少m个空间。


    2)其次待排序的元素都要在一定的范围内等等。


    桶式排序是一种分配排序。分配排序的特定是不须要进行关键码的比較。但前提是要知道待排序列的一些详细情况。


    分配排序的基本思想:说白了就是进行多次的桶式排序。


    基数排序过程无须比較keyword,而是通过“分配”和“收集”过程来实现排序。

    它们的时间复杂度可达到线性阶:O(n)。


    实例:


    扑克牌中52 张牌,可按花色和面值分成两个字段,其大小关系为:


    花色: 梅花< 方块< 红心< 黑心 
    面值: 2 < 3 < 4 < 5 < 6 < 7 < 8 < 9 < 10 < J < Q < K < A


    若对扑克牌按花色、面值进行升序排序,得到例如以下序列:





    即两张牌,若花色不同。不论面值如何,花色低的那张牌小于花色高的,仅仅有在同花色情况下。大小关系才由面值的大小确定。

    这就是多关键码排序。


    为得到排序结果,我们讨论两种排序方法。


    方法1:先对花色排序。将其分为4 个组。即梅花组、方块组、红心组、黑心组。再对每一个组分别按面值进行排序。最后。将4 个组连接起来就可以。


    方法2:先按13 个面值给出13 个编号组(2 号,3 号,...。A 号)。将牌按面值依次放入相应的编号组,分成13 堆。再按花色给出4 个编号组(梅花、方块、红心、黑心)。将2号组中牌取出分别放入相应花色组。再将3 号组中牌取出分别放入相应花色组,……,这样,4 个花色组中均按面值有序。然后。将4 个花色组依次连接起来就可以。


    设n 个元素的待排序列包括d 个关键码{k1。k2,…,kd}。则称序列对关键码{k1,k2。…,kd}有序是指:对于序列中任两个记录r[i]和r[j](1≤i≤j≤n)都满足下列有序关系:


    当中k1 称为最主位关键码,kd 称为最次位关键码 。


    两种多关键码排序方法:


    多关键码排序依照从最主位关键码到最次位关键码或从最次位到最主位关键码的顺序逐次排序,分两种方法:


    最高位优先(Most Significant Digit first)法,简称MSD 法:


    1)先按k1 排序分组,将序列分成若干子序列,同一组序列的记录中。关键码k1 相等。


    2)再对各组按k2 排序分成子组。之后。对后面的关键码继续这种排序分组,直到按最次位关键码kd 对各子组排序后。


    3)再将各组连接起来。便得到一个有序序列。

    扑克牌按花色、面值排序中介绍的方法一即是MSD 法。


    最低位优先(Least Significant Digit first)法,简称LSD 法:


    1) 先从kd 開始排序,再对kd-1进行排序。依次反复,直到按k1排序分组分成最小的子序列后。


    2) 最后将各个子序列连接起来,便可得到一个有序的序列, 扑克牌按花色、面值排序中介绍的方法二即是LSD 法。


    基于LSD方法的链式基数排序的基本思想


    “多keyword排序”的思想实现“单keyword排序”。

    对数字型或字符型的单keyword,能够看作由多个数位或多个字符构成的多keyword。此时能够採用“分配-收集”的方法进行排序,这一过程称作基数排序法,当中每一个数字或字符可能的取值个数称为基数。比方,扑克牌的花色基数为4,面值基数为13。在整理扑克牌时,既能够先按花色整理。也能够先按面值整理。按花色整理时,先按红、黑、方、花的顺序分成4摞(分配),再按此顺序再叠放在一起(收集)。然后按面值的顺序分成13摞(分配)。再按此顺序叠放在一起(收集),如此进行二次分配和收集就可以将扑克牌排列有序。


    基数排序:


    是依照低位先排序,然后收集;再依照高位排序,然后再收集。依次类推,直到最高位。

    有时候有些属性是有优先级顺序的。先按低优先级排序,再按高优先级排序。

    最后的次序就是高优先级高的在前。高优先级同样的低优先级高的在前。基数排序基于分别排序,分别收集,所以是稳定的。


    算法实现:

    Void RadixSort(Node L[],length,maxradix)

    {

    int m,n,k,lsp;

    k=1;m=1;

    int temp[10][length-1];

    Empty(temp); //清空暂时空间

    while(k<maxradix) //遍历全部keyword

    {

    for(int i=0;i<length;i++) //分配过程

    {

    if(L[i]<m)

    Temp[0][n]=L[i];

    else

    Lsp=(L[i]/m)%10; //确定keyword

    Temp[lsp][n]=L[i];

    n++;

    }

    CollectElement(L,Temp); //收集

    n=0;

    m=m*10;

    k++;

    }

    }

    总结


    各种排序的稳定性,时间复杂度和空间复杂度总结:




    我们比較时间复杂度函数的情况:




    时间复杂度函数O(n)的增长情况




    所以对n较大的排序记录。一般的选择都是时间复杂度为O(nlog2n)的排序方法。


    时间复杂度来说:


    (1)平方阶(O(n2))排序

    各类简单排序:直接插入、直接选择和冒泡排序。


    (2)线性对数阶(O(nlog2n))排序

    高速排序、堆排序和归并排序;


    (3)O(n1+§))排序,§是介于0和1之间的常数。

    希尔排序


    (4)线性阶(O(n))排序

    基数排序。此外还有桶、箱排序。


    说明:


    当原表有序或基本有序时。直接插入排序和冒泡排序将大大降低比較次数和移动记录的次数,时间复杂度可降至O(n);


    而高速排序则相反,当原表基本有序时。将蜕化为冒泡排序,时间复杂度提高为O(n2);


    原表是否有序,对简单选择排序、堆排序、归并排序和基数排序的时间复杂度影响不大。


    稳定性:


    排序算法的稳定性:若待排序的序列中,存在多个具有同样keyword的记录。经过排序, 这些记录的相对次序保持不变。则称该算法是稳定的;若经排序后,记录的相对 次序发生了改变,则称该算法是不稳定的。


    稳定性的优点:排序算法假设是稳定的。那么从一个键上排序。然后再从还有一个键上排序,第一个键排序的结果能够为第二个键排序所用。基数排序就是这样。先按低位排序。逐次按高位排序,低位同样的元素其顺序再高位也同样时是不会改变的。另外。假设排序算法稳定。能够避免多余的比較;


    稳定的排序算法:冒泡排序、插入排序、归并排序和基数排序


    不是稳定的排序算法:选择排序、高速排序、希尔排序、堆排序


    选择排序算法准则:


    每种排序算法都各有优缺点。因此,在有用时需依据不同情况适当选用,甚至能够将多种方法结合起来使用。


    选择排序算法的根据


    影响排序的因素有非常多。平均时间复杂度低的算法并不一定就是最优的。

    相反,有时平均时间复杂度高的算法可能更适合某些特殊情况。同一时候,选择算法时还得考虑它的可读性,以利于软件的维护。

    一般而言。须要考虑的因素有下面四点:


    1.待排序的记录数目n的大小。

    2.记录本身数据量的大小,也就是记录中除keyword外的其它信息量的大小;

    3.keyword的结构及其分布情况;

    4.对排序稳定性的要求。


    设待排序元素的个数为n.


    1)当n较大,则应採用时间复杂度为O(nlog2n)的排序方法:高速排序、堆排序或归并排序序。


    高速排序:是眼下基于比較的内部排序中被觉得是最好的方法,当待排序的keyword是随机分布时,高速排序的平均时间最短;


    堆排序:假设内存空间同意且要求稳定性的,


    归并排序:它有一定数量的数据移动,所以我们可能过与插入排序组合。先获得一定长度的序列。然后再合并,在效率上将有所提高。


    2)当n较大,内存空间同意。且要求稳定性 =》归并排序


    3)当n较小。可採用直接插入或直接选择排序。


    直接插入排序:当元素分布有序,直接插入排序将大大降低比較次数和移动记录的次数。


    直接选择排序:元素分布有序,假设不要求稳定性。选择直接选择排序


    5)一般不使用或不直接使用传统的冒泡排序。


    6)基数排序


    它是一种稳定的排序算法,但有一定的局限性:


    1、keyword可分解。

    2、记录的keyword位数较少,假设密集更好

    3、假设是数字时。最好是无符号的。否则将添加对应的映射复杂度,可先将其正负分开排序。



    概述


    排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序。而外部排序是因排序的数据非常大,一次不能容纳所有的排序记录,在排序过程中须要訪问外存。


    我们这里说说八大排序就是内部排序。




    当n较大。则应採用时间复杂度为O(nlog2n)的排序方法:高速排序、堆排序或归并排序序。


    高速排序:是眼下基于比較的内部排序中被觉得是最好的方法,当待排序的keyword是随机分布时,高速排序的平均时间最短。


    1.插入排序—直接插入排序(Straight Insertion Sort)


    基本思想:


    将一个记录插入到已排序好的有序表中,从而得到一个新。记录数增1的有序表。即:先将序列的第1个记录看成是一个有序的子序列,然后从第2个记录逐个进行插入,直至整个序列有序为止。


    要点:设立哨兵,作为暂时存储和推断数组边界之用。


    直接插入排序演示样例:




    假设碰见一个和插入元素相等的,那么插入元素把想插入的元素放在相等元素的后面。

    所以,相等元素的前后顺序没有改变,从原无序序列出去的顺序就是排好序后的顺序,所以插入排序是稳定的。


    算法的实现:

    .NET、void print(int a[], int n ,int i){

    cout<<i <<":";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }



    void InsertSort(int a[], int n)

    {

    for(int i= 1; i<n; i++){

    if(a[i] < a[i-1]){ //若第i个元素大于i-1元素。直接插入。小于的话,移动有序表后插入

    int j= i-1;

    int x = a[i]; //复制为哨兵,即存储待排序元素

    a[i] = a[i-1]; //先后移一个元素

    while(x < a[j]){ //查找在有序表的插入位置

    a[j+1] = a[j];

    j--; //元素后移

    }

    a[j+1] = x; //插入到正确位置

    }

    print(a,n,i); //打印每趟排序的结果

    }


    }


    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    InsertSort(a,8);

    print(a,8,8);

    }

    效率:


    时间复杂度:O(n^2).


    其它的插入排序有二分插入排序。2-路插入排序。


    2.插入排序—希尔排序(Shell`s Sort)


    希尔排序是1959 年由D.L.Shell 提出来的,相对直接排序有较大的改进。希尔排序又叫缩小增量排序


    基本思想:


    先将整个待排序的记录序列切割成为若干子序列分别进行直接插入排序,待整个序列中的记录“基本有序”时,再对全体记录进行依次直接插入排序。


    操作方法:


    1.选择一个增量序列t1,t2。…。tk。当中ti>tj,tk=1;

    2.按增量序列个数k。对序列进行k 趟排序。

    3.每趟排序。依据相应的增量ti,将待排序列切割成若干长度为m 的子序列。分别对各子表进行直接插入排序。仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。


    希尔排序的演示样例:




    算法实现:


    我们简单处理增量序列:增量序列d = {n/2 ,n/4, n/8 .....1} n为要排序数的个数


    即:先将要排序的一组记录按某个增量d(n/2,n为要排序数的个数)分成若干组子序列,每组中记录的下标相差d.对每组中所有元素进行直接插入排序,然后再用一个较小的增量(d/2)对它进行分组,在每组中再进行直接插入排序。继续不断缩小增量直至为1,最后使用直接插入排序完毕排序。

    void print(int a[], int n ,int i){

    cout<<i <<":";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }

    /**

    * 直接插入排序的一般形式

    *

    * @param int dk 缩小增量,假设是直接插入排序。dk=1

    *

    */


    void ShellInsertSort(int a[], int n, int dk)

    {

    for(int i= dk; i<n; ++i){

    if(a[i] < a[i-dk]){ //若第i个元素大于i-1元素,直接插入。小于的话,移动有序表后插入

    int j = i-dk;

    int x = a[i]; //复制为哨兵,即存储待排序元素

    a[i] = a[i-dk]; //首先后移一个元素

    while(x < a[j]){ //查找在有序表的插入位置

    a[j+dk] = a[j];

    j -= dk; //元素后移

    }

    a[j+dk] = x; //插入到正确位置

    }

    print(a, n,i );

    }


    }


    /**

    * 先按增量d(n/2,n为要排序数的个数进行希尔排序

    *

    */

    void shellSort(int a[], int n){


    int dk = n/2;

    while( dk >= 1 ){

    ShellInsertSort(a, n, dk);

    dk = dk/2;

    }

    }

    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    //ShellInsertSort(a,8,1); //直接插入排序

    shellSort(a,8); //希尔插入排序

    print(a,8,8);

    }

    希尔排序时效分析非常难,关键码的比較次数与记录移动次数依赖于增量因子序列d的选取,特定情况下能够准确估算出关键码的比較次数和记录的移动次数。眼下还没有人给出选取最好的增量因子序列的方法。增量因子序列能够有各种取法。有取奇数的。也有取质数的。但须要注意:增量因子中除1 外没有公因子,且最后一个增量因子必须为1。希尔排序方法是一个不稳定的排序方法。


    3.选择排序—简单选择排序(Simple Selection Sort)


    基本思想:


    在要排序的一组数中,选出最小(或者最大)的一个数与第1个位置的数交换;然后在剩下的数其中再找最小(或者最大)的与第2个位置的数交换,依次类推,直到第n-1个元素(倒数第二个数)和第n个元素(最后一个数)比較为止。


    简单选择排序的演示样例:



    操作方法:


    第一趟,从n 个记录中找出关键码最小的记录与第一个记录交换;

    第二趟。从第二个记录開始的n-1 个记录中再选出关键码最小的记录与第二个记录交换;


    以此类推.....

    第i 趟,则从第i 个记录開始的n-i+1 个记录中选出关键码最小的记录与第i 个记录交换。


    直到整个序列按关键码有序。


    算法实现:

    void print(int a[], int n ,int i){

    cout<<"第"<<i+1 <<"趟 : ";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }

    /**

    * 数组的最小值

    *

    * @return int 数组的键值

    */

    int SelectMinKey(int a[], int n, int i)

    {

    int k = i;

    for(int j=i+1 ;j< n; ++j) {

    if(a[k] > a[j]) k = j;

    }

    return k;

    }


    /**

    * 选择排序

    *

    */

    void selectSort(int a[], int n){

    int key, tmp;

    for(int i = 0; i< n; ++i) {

    key = SelectMinKey(a, n,i); //选择最小的元素

    if(key != i){

    tmp = a[i]; a[i] = a[key]; a[key] = tmp; //最小元素与第i位置元素互换

    }

    print(a, n , i);

    }

    }

    int main(){

    int a[8] = {3,1,5,7,2,4,9,6};

    cout<<"初始值:";

    for(int j= 0; j<8; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl<<endl;

    selectSort(a, 8);

    print(a,8,8);

    }

    简单选择排序的改进——二元选择排序


    简单选择排序。每趟循环仅仅能确定一个元素排序后的定位。我们能够考虑改进为每趟循环确定两个元素(当前趟最大和最小记录)的位置,从而降低排序所需的循环次数。改进后对n个数据进行排序,最多仅仅需进行[n/2]趟循环就可以。

    详细实现例如以下:

    void SelectSort(int r[],int n) {

    int i ,j , min ,max, tmp;

    for (i=1 ;i <= n/2;i++) {

    // 做不超过n/2趟选择排序

    min = i; max = i ; //分别记录最大和最小keyword记录位置

    for (j= i+1; j<= n-i; j++) {

    if (r[j] > r[max]) {

    max = j ; continue ;

    }

    if (r[j]< r[min]) {

    min = j ;

    }

    }

    //该交换操作还可分情况讨论以提高效率

    tmp = r[i-1]; r[i-1] = r[min]; r[min] = tmp;

    tmp = r[n-i]; r[n-i] = r[max]; r[max] = tmp;


    }

    }

    4.选择排序—堆排序(Heap Sort)


    堆排序是一种树形选择排序。是对直接选择排序的有效改进。


    基本思想:


    堆的定义例如以下:具有n个元素的序列(k1,k2,...,kn),当且仅当满足




    时称之为堆。由堆的定义能够看出,堆顶元素(即第一个元素)必为最小项(小顶堆)。


    若以一维数组存储一个堆,则堆相应一棵全然二叉树,且全部非叶结点的值均不大于(或不小于)其子女的值,根结点(堆顶元素)的值是最小(或最大)的。

    如:


    (a)大顶堆序列:(96, 83,27,38,11,09)

    (b) 小顶堆序列:(12,36。24,85,47。30,53,91)




    初始时把要排序的n个数的序列看作是一棵顺序存储的二叉树(一维数组存储二叉树),调整它们的存储序。使之成为一个堆。将堆顶元素输出,得到n 个元素中最小(或最大)的元素,这时堆的根节点的数最小(或者最大)。然后对前面(n-1)个元素又一次调整使之成为堆,输出堆顶元素,得到n 个元素中次小(或次大)的元素。依此类推,直到仅仅有两个节点的堆,并对它们作交换,最后得到有n个节点的有序序列。称这个过程为堆排序。


    因此,实现堆排序需解决两个问题:


    1. 怎样将n 个待排序的数建成堆;
    2. 输出堆顶元素后,如何调整剩余n-1 个元素。使其成为一个新堆。


    首先讨论第二个问题:输出堆顶元素后。对剩余n-1元素又一次建成堆的调整过程。


    调整小顶堆的方法:


    1)设有m 个元素的堆,输出堆顶元素后。剩下m-1 个元素。将堆底元素送入堆顶((最后一个元素与堆顶进行交换),堆被破坏,其原因仅是根结点不满足堆的性质。


    2)将根结点与左、右子树中较小元素的进行交换。


    3)若与左子树交换:假设左子树堆被破坏,即左子树的根结点不满足堆的性质。则反复方法 (2).


    4)若与右子树交换。假设右子树堆被破坏,即右子树的根结点不满足堆的性质。则反复方法 (2).


    5)继续对不满足堆性质的子树进行上述交换操作。直到叶子结点,堆被建成。


    称这个自根结点到叶子结点的调整过程为筛选。如图:




    再讨论对n 个元素初始建堆的过程。


    建堆方法:对初始序列建堆的过程,就是一个重复进行筛选的过程。


    1)n 个结点的全然二叉树,则最后一个结点是第个结点的子树。

    2)筛选从第个结点为根的子树開始。该子树成为堆。

    3)之后向前依次对各结点为根的子树进行筛选,使之成为堆,直到根结点。


    如图建堆初始过程:无序序列:(49。38,65,97,76,13。27。49)





    算法的实现:


    从算法描写叙述来看,堆排序须要两个过程。一是建立堆,二是堆顶与堆的最后一个元素交换位置。所以堆排序有两个函数组成。一是建堆的渗透函数。二是重复调用渗透函数实现排序的函数。

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }




    /**

    * 已知H[s…m]除了H[s] 外均满足堆的定义

    * 调整H[s],使其成为大顶堆.即将对第s个结点为根的子树筛选,

    *

    * @param H是待调整的堆数组

    * @param s是待调整的数组元素的位置

    * @param length是数组的长度

    *

    */

    void HeapAdjust(int H[],int s, int length)

    {

    int tmp = H[s];

    int child = 2*s+1; //左孩子结点的位置。(i+1 为当前调整结点的右孩子结点的位置)

    while (child < length) {

    if(child+1 <length && H[child]<H[child+1]) { // 假设右孩子大于左孩子(找到比当前待调整结点大的孩子结点)

    ++child ;

    }

    if(H[s]<H[child]) { // 假设较大的子结点大于父结点

    H[s] = H[child]; // 那么把较大的子结点往上移动。替换它的父结点

    s = child; // 又一次设置s ,即待调整的下一个结点的位置

    child = 2*s+1;

    } else { // 假设当前待调整结点大于它的左右孩子,则不须要调整。直接退出

    break;

    }

    H[s] = tmp; // 当前待调整的结点放到比其大的孩子结点位置上

    }

    print(H,length);

    }



    /**

    * 初始堆进行调整

    * 将H[0..length-1]建成堆

    * 调整完之后第一个元素是序列的最小的元素

    */

    void BuildingHeap(int H[], int length)

    {

    //最后一个有孩子的节点的位置 i= (length -1) / 2

    for (int i = (length -1) / 2 ; i >= 0; --i)

    HeapAdjust(H,i,length);

    }

    /**

    * 堆排序算法

    */

    void HeapSort(int H[],int length)

    {

    //初始堆

    BuildingHeap(H, length);

    //从最后一个元素開始对序列进行调整

    for (int i = length - 1; i > 0; --i)

    {

    //交换堆顶元素H[0]和堆中最后一个元素

    int temp = H[i]; H[i] = H[0]; H[0] = temp;

    //每次交换堆顶元素和堆中最后一个元素之后。都要对堆进行调整

    HeapAdjust(H,0,i);

    }

    }


    int main(){

    int H[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(H,10);

    HeapSort(H,10);

    //selectSort(a, 8);

    cout<<"结果:";

    print(H,10);


    }

    分析:


    设树深度为k,

    从根到叶的筛选,元素比較次数至多2(k-1)次。交换记录至多k 次。

    所以。在建好堆后,排序过程中的筛选次数不超过下式:



    而建堆时的比較次数不超过4n 次,因此堆排序最坏情况下,时间复杂度也为:O(nlogn )。


    5.交换排序—冒泡排序(Bubble Sort)


    基本思想:


    在要排序的一组数中,对当前还未排好序的范围内的所有数,自上而下对相邻的两个数依次进行比較和调整,让较大的数往下沉,较小的往上冒。即:每当两相邻的数比較后发现它们的排序与排序要求相反时,就将它们互换。


    冒泡排序的演示样例:




    算法的实现:

    void bubbleSort(int a[], int n){

    for(int i =0 ; i< n-1; ++i) {

    for(int j = 0; j < n-i-1; ++j) {

    if(a[j] > a[j+1])

    {

    int tmp = a[j] ; a[j] = a[j+1] ; a[j+1] = tmp;

    }

    }

    }

    }

    冒泡排序算法的改进


    对冒泡排序常见的改进方法是增加一标志性变量exchange,用于标志某一趟排序过程中是否有数据交换,假设进行某一趟排序时并没有进行数据交换,则说明数据已经按要求排列好,可马上结束排序,避免不必要的比較过程。本文再提供下面两种改进算法:


    1.设置一标志性变量pos,用于记录每趟排序中最后一次进行交换的位置。因为pos位置之后的记录均已交换到位,故在进行下一趟排序时仅仅要扫描到pos位置就可以。


    改进后算法例如以下:

    void Bubble_1 ( int r[], int n) {

    int i= n -1; //初始时,最后位置保持不变

    while ( i> 0) {

    int pos= 0; //每趟開始时,无记录交换

    for (int j= 0; j< i; j++)

    if (r[j]> r[j+1]) {

    pos= j; //记录交换的位置

    int tmp = r[j]; r[j]=r[j+1];r[j+1]=tmp;

    }

    i= pos; //为下一趟排序作准备

    }

    }

    2.传统冒泡排序中每一趟排序操作仅仅能找到一个最大值或最小值,我们考虑利用在每趟排序中进行正向和反向两遍冒泡的方法一次能够得到两个终于值(最大者和最小者) , 从而使排序趟数差点儿降低了一半。


    改进后的算法实现为:

    void Bubble_2 ( int r[], int n){

    int low = 0;

    int high= n -1; //设置变量的初始值

    int tmp,j;

    while (low < high) {

    for (j= low; j< high; ++j) //正向冒泡,找到最大者

    if (r[j]> r[j+1]) {

    tmp = r[j]; r[j]=r[j+1];r[j+1]=tmp;

    }

    --high; //改动high值, 前移一位

    for ( j=high; j>low; --j) //反向冒泡,找到最小者

    if (r[j]<r[j-1]) {

    tmp = r[j]; r[j]=r[j-1];r[j-1]=tmp;

    }

    ++low; //改动low值,后移一位

    }

    }

    6.交换排序—高速排序(Quick Sort)


    基本思想:


    1)选择一个基准元素,通常选择第一个元素或者最后一个元素,

    2)通过一趟排序讲待排序的记录切割成独立的两部分。当中一部分记录的元素值均比基准元素值小。

    还有一部分记录的 元素值比基准值大。

    3)此时基准元素在其排好序后的正确位置

    4)然后分别对这两部分记录用相同的方法继续进行排序。直到整个序列有序。


    高速排序的演示样例:


    (a)一趟排序的过程:




    (b)排序的全过程




    算法的实现:


    递归实现:

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    void swap(int *a, int *b)

    {

    int tmp = *a;

    *a = *b;

    *b = tmp;

    }


    int partition(int a[], int low, int high)

    {

    int privotKey = a[low]; //基准元素

    while(low < high){ //从表的两端交替地向中间扫描

    while(low < high && a[high] >= privotKey) --high; //从high 所指位置向前搜索,至多到low+1 位置。将比基准元素小的交换到低端

    swap(&a[low], &a[high]);

    while(low < high && a[low] <= privotKey ) ++low;

    swap(&a[low], &a[high]);

    }

    print(a,10);

    return low;

    }



    void quickSort(int a[], int low, int high){

    if(low < high){

    int privotLoc = partition(a, low, high); //将表一分为二

    quickSort(a, low, privotLoc -1); //递归对低子表递归排序

    quickSort(a, privotLoc + 1, high); //递归对高子表递归排序

    }

    }


    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(a,10);

    quickSort(a,0,9);

    cout<<"结果:";

    print(a,10);


    }

    分析:


    高速排序是通常被觉得在同数量级(O(nlog2n))的排序方法中平均性能最好的。但若初始序列按关键码有序或基本有序时。快排序反而蜕化为冒泡排序。为改进之。通常以“三者取中法”来选取基准记录,即将排序区间的两个端点与中点三个记录关键码居中的调整为支点记录。

    高速排序是一个不稳定的排序方法。


    高速排序的改进


    在本改进算法中,仅仅对长度大于k的子序列递归调用高速排序,让原序列基本有序,然后再对整个基本有序序列用插入排序算法排序。

    实践证明,改进后的算法时间复杂度有所减少,且当k取值为 8 左右时,改进算法的性能最佳。算法思想例如以下:

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    void swap(int *a, int *b)

    {

    int tmp = *a;

    *a = *b;

    *b = tmp;

    }


    int partition(int a[], int low, int high)

    {

    int privotKey = a[low]; //基准元素

    while(low < high){ //从表的两端交替地向中间扫描

    while(low < high && a[high] >= privotKey) --high; //从high 所指位置向前搜索,至多到low+1 位置。将比基准元素小的交换到低端

    swap(&a[low], &a[high]);

    while(low < high && a[low] <= privotKey ) ++low;

    swap(&a[low], &a[high]);

    }

    print(a,10);

    return low;

    }



    void qsort_improve(int r[ ],int low,int high, int k){

    if( high -low > k ) { //长度大于k时递归, k为指定的数

    int pivot = partition(r, low, high); // 调用的Partition算法保持不变

    qsort_improve(r, low, pivot - 1,k);

    qsort_improve(r, pivot + 1, high,k);

    }

    }

    void quickSort(int r[], int n, int k){

    qsort_improve(r,0,n,k);//先调用改进算法Qsort使之基本有序


    //再用插入排序对基本有序序列排序

    for(int i=1; i<=n;i ++){

    int tmp = r[i];

    int j=i-1;

    while(tmp < r[j]){

    r[j+1]=r[j]; j=j-1;

    }

    r[j+1] = tmp;

    }


    }




    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    cout<<"初始值:";

    print(a,10);

    quickSort(a,9,4);

    cout<<"结果:";

    print(a,10);


    }

    7.归并排序(Merge Sort)


    基本思想:


    归并(Merge)排序法是将两个(或两个以上)有序表合并成一个新的有序表,即把待排序序列分为若干个子序列,每一个子序列是有序的。然后再把有序子序列合并为总体有序序列。


    归并排序演示样例:




    合并方法:


    设r[i…n]由两个有序子表r[i…m]和r[m+1…n]组成,两个子表长度分别为n-i +1、n-m。


    1.j=m+1。k=i。i=i; //置两个子表的起始下标及辅助数组的起始下标


    2.若i>m 或j>n,转⑷ //当中一个子表已合并完,比較选取结束


    3.//选取r[i]和r[j]较小的存入辅助数组rf

    假设r[i]<r[j],rf[k]=r[i]; i++; k++; 转⑵

    否则,rf[k]=r[j]。 j++。 k++; 转⑵


    4.//将尚未处理完的子表中元素存入rf

    假设i<=m,将r[i…m]存入rf[k…n] //前一子表非空
    假设j<=n , 将r[j…n] 存入rf[k…n] //后一子表非空


    5.合并结束。

    //将r[i…m]和r[m +1 …n]归并到辅助数组rf[i…n]

    void Merge(ElemType *r,ElemType *rf, int i, int m, int n)

    {

    int j,k;

    for(j=m+1,k=i; i<=m && j <=n ; ++k){

    if(r[j] < r[i]) rf[k] = r[j++];

    else rf[k] = r[i++];

    }

    while(i <= m) rf[k++] = r[i++];

    while(j <= n) rf[k++] = r[j++];

    }

    归并的迭代算法


    1个元素的表总是有序的。所以对n 个元素的待排序列,每一个元素可看成1 个有序子表。对子表两两合并生成n/2个子表,所得子表除最后一个子表长度可能为1 外。其余子表长度均为2。

    再进行两两合并,直到生成n 个元素按关键码有序的表。

    void print(int a[], int n){

    for(int j= 0; j<n; j++){

    cout<<a[j] <<" ";

    }

    cout<<endl;

    }


    //将r[i…m]和r[m +1 …n]归并到辅助数组rf[i…n]

    void Merge(ElemType *r,ElemType *rf, int i, int m, int n)

    {

    int j,k;

    for(j=m+1,k=i; i<=m && j <=n ; ++k){

    if(r[j] < r[i]) rf[k] = r[j++];

    else rf[k] = r[i++];

    }

    while(i <= m) rf[k++] = r[i++];

    while(j <= n) rf[k++] = r[j++];

    print(rf,n+1);

    }


    void MergeSort(ElemType *r, ElemType *rf, int lenght)

    {

    int len = 1;

    ElemType *q = r ;

    ElemType *tmp ;

    while(len < lenght) {

    int s = len;

    len = 2 * s ;

    int i = 0;

    while(i+ len <lenght){

    Merge(q, rf, i, i+ s-1, i+ len-1 ); //对等长的两个子表合并

    i = i+ len;

    }

    if(i + s < lenght){

    Merge(q, rf, i, i+ s -1, lenght -1); //对不等长的两个子表合并

    }

    tmp = q; q = rf; rf = tmp; //交换q,rf。以保证下一趟归并时。仍从q 归并到rf

    }

    }



    int main(){

    int a[10] = {3,1,5,7,2,4,9,6,10,8};

    int b[10];

    MergeSort(a, b, 10);

    print(b,10);

    cout<<"结果:";

    print(a,10);


    }

    两路归并的递归算法

    void MSort(ElemType *r, ElemType *rf,int s, int t)

    {

    ElemType *rf2;

    if(s==t) r[s] = rf[s];

    else

    {

    int m=(s+t)/2; /*平分*p 表*/

    MSort(r, rf2, s, m); /*递归地将p[s…m]归并为有序的p2[s…m]*/

    MSort(r, rf2, m+1, t); /*递归地将p[m+1…t]归并为有序的p2[m+1…t]*/

    Merge(rf2, rf, s, m+1,t); /*将p2[s…m]和p2[m+1…t]归并到p1[s…t]*/

    }

    }

    void MergeSort_recursive(ElemType *r, ElemType *rf, int n)

    { /*对顺序表*p 作归并排序*/

    MSort(r, rf,0, n-1);

    }

    8.桶排序/基数排序(Radix Sort)


    说基数排序之前。我们先说桶排序:


    基本思想:是将阵列分到有限数量的桶子里。每一个桶子再个别排序(有可能再使用别的排序算法或是以递回方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。

    当要被排序的阵列内的数值是均匀分配的时候,桶排序使用线性时间(Θ(n))。但桶排序并非 比較排序。他不受到 O(n log n) 下限的影响。


    简单来说。就是把数据分组,放在一个个的桶中,然后对每一个桶里面的在进行排序。


    比如要对大小为[1..1000]范围内的n个整数A[1..n]排序


    首先,能够把桶设为大小为10的范围。详细而言,设集合B[1]存储[1..10]的整数,集合B[2]存储 (10..20]的整数,……集合B[i]存储( (i-1)*10, i*10]的整数,i = 1,2,..100。总共同拥有 100个桶。


    然后。对A[1..n]从头到尾扫描一遍,把每一个A[i]放入相应的桶B[j]中。 再对这100个桶中每一个桶里的数字排序,这时可用冒泡。选择,乃至快排,一般来说任 何排序法都能够。


    最后,依次输出每一个桶里面的数字。且每一个桶中的数字从小到大输出,这 样就得到全部数字排好序的一个序列了。


    假设有n个数字,有m个桶。假设数字是平均分布的。则每一个桶里面平均有n/m个数字。

    假设对每一个桶中的数字採用高速排序,那么整个算法的复杂度是 O(n + m * n/m*log(n/m)) = O(n + nlogn - nlogm) 从上式看出。当m接近n的时候,桶排序复杂度接近O(n)


    当然。以上复杂度的计算是基于输入的n个数字是平均分布这个如果的。这个如果是非常强的 ,实际应用中效果并没有这么好。

    如果全部的数字都落在同一个桶中,那就退化成一般的排序了。


    前面说的几大排序算法 。大部分时间复杂度都是O(n2),也有部分排序算法时间复杂度是O(nlogn)。而桶式排序却能实现O(n)的时间复杂度。但桶排序的缺点是:


    1)首先是空间复杂度比較高。须要的额外开销大。排序有两个数组的空间开销,一个存放待排序数组。一个就是所谓的桶。比方待排序值是从0到m-1,那就须要m个桶,这个桶数组就要至少m个空间。


    2)其次待排序的元素都要在一定的范围内等等。


    桶式排序是一种分配排序。分配排序的特定是不须要进行关键码的比較。但前提是要知道待排序列的一些详细情况。


    分配排序的基本思想:说白了就是进行多次的桶式排序。


    基数排序过程无须比較keyword。而是通过“分配”和“收集”过程来实现排序。它们的时间复杂度可达到线性阶:O(n)。


    实例:


    扑克牌中52 张牌,可按花色和面值分成两个字段,其大小关系为:


    花色: 梅花< 方块< 红心< 黑心 
    面值: 2 < 3 < 4 < 5 < 6 < 7 < 8 < 9 < 10 < J < Q < K < A


    若对扑克牌按花色、面值进行升序排序,得到例如以下序列:





    即两张牌。若花色不同,不论面值如何。花色低的那张牌小于花色高的,仅仅有在同花色情况下。大小关系才由面值的大小确定。

    这就是多关键码排序。


    为得到排序结果,我们讨论两种排序方法。


    方法1:先对花色排序,将其分为4 个组。即梅花组、方块组、红心组、黑心组。

    再对每一个组分别按面值进行排序。最后。将4 个组连接起来就可以。


    方法2:先按13 个面值给出13 个编号组(2 号。3 号,...,A 号)。将牌按面值依次放入相应的编号组,分成13 堆。

    再按花色给出4 个编号组(梅花、方块、红心、黑心),将2号组中牌取出分别放入相应花色组,再将3 号组中牌取出分别放入相应花色组。……,这样,4 个花色组中均按面值有序,然后。将4 个花色组依次连接起来就可以。


    设n 个元素的待排序列包括d 个关键码{k1,k2,…,kd}。则称序列对关键码{k1,k2。…。kd}有序是指:对于序列中任两个记录r[i]和r[j](1≤i≤j≤n)都满足下列有序关系:


    当中k1 称为最主位关键码,kd 称为最次位关键码 。


    两种多关键码排序方法:


    多关键码排序依照从最主位关键码到最次位关键码或从最次位到最主位关键码的顺序逐次排序。分两种方法:


    最高位优先(Most Significant Digit first)法,简称MSD 法:


    1)先按k1 排序分组。将序列分成若干子序列,同一组序列的记录中。关键码k1 相等。


    2)再对各组按k2 排序分成子组。之后,对后面的关键码继续这种排序分组。直到按最次位关键码kd 对各子组排序后。


    3)再将各组连接起来,便得到一个有序序列。扑克牌按花色、面值排序中介绍的方法一即是MSD 法。


    最低位优先(Least Significant Digit first)法。简称LSD 法:


    1) 先从kd 開始排序。再对kd-1进行排序。依次反复,直到按k1排序分组分成最小的子序列后。


    2) 最后将各个子序列连接起来,便可得到一个有序的序列, 扑克牌按花色、面值排序中介绍的方法二即是LSD 法。


    基于LSD方法的链式基数排序的基本思想


    “多keyword排序”的思想实现“单keyword排序”。

    对数字型或字符型的单keyword,能够看作由多个数位或多个字符构成的多keyword,此时能够採用“分配-收集”的方法进行排序。这一过程称作基数排序法,当中每一个数字或字符可能的取值个数称为基数。比方,扑克牌的花色基数为4。面值基数为13。在整理扑克牌时,既能够先按花色整理,也能够先按面值整理。按花色整理时。先按红、黑、方、花的顺序分成4摞(分配),再按此顺序再叠放在一起(收集)。然后按面值的顺序分成13摞(分配),再按此顺序叠放在一起(收集)。如此进行二次分配和收集就可以将扑克牌排列有序。


    基数排序:


    是依照低位先排序。然后收集;再依照高位排序。然后再收集;依次类推。直到最高位。

    有时候有些属性是有优先级顺序的,先按低优先级排序。再按高优先级排序。

    最后的次序就是高优先级高的在前。高优先级同样的低优先级高的在前。

    基数排序基于分别排序,分别收集。所以是稳定的。


    算法实现:

    Void RadixSort(Node L[],length,maxradix)

    {

    int m,n,k,lsp;

    k=1;m=1;

    int temp[10][length-1];

    Empty(temp); //清空暂时空间

    while(k<maxradix) //遍历全部keyword

    {

    for(int i=0;i<length;i++) //分配过程

    {

    if(L[i]<m)

    Temp[0][n]=L[i];

    else

    Lsp=(L[i]/m)%10; //确定keyword

    Temp[lsp][n]=L[i];

    n++;

    }

    CollectElement(L,Temp); //收集

    n=0;

    m=m*10;

    k++;

    }

    }

    总结


    各种排序的稳定性,时间复杂度和空间复杂度总结:




    我们比較时间复杂度函数的情况:




    时间复杂度函数O(n)的增长情况




    所以对n较大的排序记录。一般的选择都是时间复杂度为O(nlog2n)的排序方法。


    时间复杂度来说:


    (1)平方阶(O(n2))排序

    各类简单排序:直接插入、直接选择和冒泡排序;


    (2)线性对数阶(O(nlog2n))排序

    高速排序、堆排序和归并排序;


    (3)O(n1+§))排序,§是介于0和1之间的常数。

    希尔排序


    (4)线性阶(O(n))排序

    基数排序,此外还有桶、箱排序。


    说明:


    当原表有序或基本有序时。直接插入排序和冒泡排序将大大降低比較次数和移动记录的次数。时间复杂度可降至O(n);


    而高速排序则相反。当原表基本有序时,将蜕化为冒泡排序,时间复杂度提高为O(n2);


    原表是否有序,对简单选择排序、堆排序、归并排序和基数排序的时间复杂度影响不大。


    稳定性:


    排序算法的稳定性:若待排序的序列中,存在多个具有同样keyword的记录。经过排序, 这些记录的相对次序保持不变。则称该算法是稳定的。若经排序后,记录的相对 次序发生了改变,则称该算法是不稳定的。


    稳定性的优点:排序算法假设是稳定的。那么从一个键上排序,然后再从还有一个键上排序,第一个键排序的结果能够为第二个键排序所用。

    基数排序就是这样,先按低位排序,逐次按高位排序,低位同样的元素其顺序再高位也同样时是不会改变的。另外,假设排序算法稳定。能够避免多余的比較;


    稳定的排序算法:冒泡排序、插入排序、归并排序和基数排序


    不是稳定的排序算法:选择排序、高速排序、希尔排序、堆排序


    选择排序算法准则:


    每种排序算法都各有优缺点。因此,在有用时需依据不同情况适当选用,甚至能够将多种方法结合起来使用。


    选择排序算法的根据


    影响排序的因素有非常多,平均时间复杂度低的算法并不一定就是最优的。相反。有时平均时间复杂度高的算法可能更适合某些特殊情况。同一时候,选择算法时还得考虑它的可读性,以利于软件的维护。一般而言,须要考虑的因素有下面四点:


    1.待排序的记录数目n的大小。

    2.记录本身数据量的大小。也就是记录中除keyword外的其它信息量的大小;

    3.keyword的结构及其分布情况;

    4.对排序稳定性的要求。


    设待排序元素的个数为n.


    1)当n较大,则应採用时间复杂度为O(nlog2n)的排序方法:高速排序、堆排序或归并排序序。


    高速排序:是眼下基于比較的内部排序中被觉得是最好的方法。当待排序的keyword是随机分布时。高速排序的平均时间最短。


    堆排序:假设内存空间同意且要求稳定性的,


    归并排序:它有一定数量的数据移动。所以我们可能过与插入排序组合,先获得一定长度的序列。然后再合并。在效率上将有所提高。


    2)当n较大。内存空间同意。且要求稳定性 =》归并排序


    3)当n较小,可採用直接插入或直接选择排序。


    直接插入排序:当元素分布有序。直接插入排序将大大降低比較次数和移动记录的次数。


    直接选择排序:元素分布有序,假设不要求稳定性,选择直接选择排序


    5)一般不使用或不直接使用传统的冒泡排序。


    6)基数排序


  • 相关阅读:
    队列加分项(选作,根据博客质量加1-5分)(补博客)
    队列课下作业(补博客)
    20162319 2017-2018-20162319 《程序设计与数据结构》第7周学习总结
    20162319 补博客——用数组实现循环队列 06.第六周
    20162319 补博客——排序课下作业
    20162319 2017-2018-20162319 《程序设计与数据结构》第5周学习总结
    20162319 2017-2018-20162319 《程序设计与数据结构》第6周学习总结
    20162319 第二学期第一次实验:线性表的应用
    20162314 Experiment 2
    20162314 《Program Design & Data Structures》Learning Summary Of The Seventh Week
  • 原文地址:https://www.cnblogs.com/yjbjingcha/p/6939945.html
Copyright © 2011-2022 走看看