数据结构-排序二

291 阅读7分钟

开场白

这篇主要介绍归并排序快速排序

归并排序

归并排序(Merging Sort) 就是利⽤归并的思想实现排序⽅法. 它的原理是假设初始序 列含有n个记录,则可以看成n个有序的⼦序列. 每个⼦序列的⻓度为1,然后两两合并.得 到[n/2]个⻓度为2或1的有序⼦序列, 再两两归并.......如此重复,直到得到⼀个⻓度为n 的有序序列为此. 这种排序⽅法称为2路归并排序。

示例模拟排序流程

r = [50,10,90,30,70,40,80,60,20] length = 9

  1. 获取中间下标,将数组分成两部分:[50,10,90,30,70] , [40,80,60,20]
  2. 讲子数组再进行分割:[50,10,90],[30,70] | [40,80],[60,20]
  3. 同理:[50,10],[90]|[30],[70] || [40], [80]|[60],[20]
  4. 最后结果:[50],[10]|[90] || [30]|[70] ||| [40]|[80] || [60]|[20]
  5. 按照上放的分割同级比较,进行排序
  6. [10,50],[90]|[30],[70] || [40], [80]|[60],[20]
  7. [10,50,90],[30,70]|[40,80]|[20,60]
  8. [10,30,50,70,90],[20,40,60,80]
  9. [10,20,30,40,50,60,70,80,90]

如图:

代码

1.递归方式
//合并函数,i代表低位,n代表高位,m代表中间位置
void Merge(int SR[], int TR[], int i, int m, int n) {
    int j, k;
    //SR分成两部分[i,m]和[m+1,n],两部分中元素依次比较,较小的放入TR中
    for (k = i, j = m+1; i<=m && j<=n; k++) {
        if (SR[i]<SR[j]) {
            TR[k]=SR[i++];
        } else {
            TR[k]=SR[j++];
        }
    }
    
    //[i,m]部分没有结束,把未插入的元素插入到TR中
    if (i <= m) {
        for (int l = 0; l<= m-i; l++) {
            TR[k+l]=SR[i+l];
        }
    }
    
    //[m+1,n]部分没有结束,把未插入的元素插入到TR中
    if (j <= n) {
        for (int l = 0; l <= n-j; l++) {
            TR[k+l]=SR[j+l];
        }
    }
}

//SR中low-high的元素进行归并排序
void MSort(int SR[], int TR1[], int low, int high) {
    if (low == high) {//递归出口 递归到SR中只剩下一个元素的时候
        TR1[low] = SR[low];
    } else {
        //去中间分隔元素下标
        int mid = (low + high) / 2;
        //创建TR2数组
        int TR2[MAXSIZE +1];
        
        //类似处理二叉树左子树的形式,递归进行SR中low-mid中的元素进行归并排序
        MSort(SR, TR2, low, mid);
        //类似处理二叉树右子树的形式,递归进行SR中(mid+1)-high中的元素进行归并排序
        MSort(SR, TR2, mid + 1, high);
        //将TR2中元素进行排序,将结果放到TR1中
        Merge(TR2, TR1, low, mid, high);
    }
}

//归并排序 入口
void MergeSort(SqList *L) {
    MSort(L->r, L->r, 1, L->length);
}
2.非递归方式
void MergePass(int SR[], int TR[], int s, int length) {
    int i = 1;
    
    //①合并数组
    //s=1 循环结束位置:8 (9-2*1+1=8)
    //s=2 循环结束位置:6 (9-2*2+1=6)
    //s=4 循环结束位置:2 (9-2*4+1=2)
    //s=8 循环结束位置:-6(9-2*8+1=-6) s = 8时,不会进入到循环;
    while (i <= length-2*s+1) {
        //两两归并(合并相邻的2段数据)
        Merge(SR, TR, i, i+s-1, i+2*s-1);
        i = i+2*s;
        /*
         s = 1,i = 1,Merge(SR,TR,1,1,2);
         s = 1,i = 3,Merge(SR,TR,3,3,4);
         s = 1,i = 5,Merge(SR,TR,5,5,6);
         s = 1,i = 7,Merge(SR,TR,7,7,8);
         s = 1,i = 9,退出循环;
         */
        
        /*
         s = 2,i = 1,Merge(SR,TR,1,2,4);
         s = 2,i = 5,Merge(SR,TR,5,6,8);
         s = 2,i = 9,退出循环;
         */
        
        /*
         s = 4,i = 1,Merge(SR,TR,1,4,8);
         s = 4,i = 9,退出循环;
         */
    }
    
    //②如果i<length-s+1,表示有2个长度不等的子序列. 其中一个长度为length,另一个小于length
    // 1 < (9-8+1)(2)
    //s = 8时, 1 < (9-8+1)
    if (i < length-s+1) {
        //Merge(SR,TR,1,8,9)
        Merge(SR, TR, i, i+s-1, length);
    } else {//③只剩下一个子序列;
        for (int j = i; j <= length; j++) {
            TR[j] = SR[j];
        }
    }
}

void MergeSort2(SqList *L) {
    int *TR = (int *)malloc(sizeof(int) * L->length);
    int k = 1;
    //k的拆分变换是 1,2,4,8;
    while (k < L->length) {
        //将SR数组按照s=2的长度进行拆分合并,结果存储到TR数组中;
        //注意:此时经过第一轮的归并排序的结果是存储到TR数组了;
        MergePass(L->r, TR, k, L->length);
        k = 2*k;
        //将刚刚归并排序后的TR数组,按照s = 2k的长度进行拆分合并. 结果存储到L->r数组中;
        //注意:因为上一轮的排序的结果是存储到TR数组,所以这次排序的数据应该是再次对TR数组排序;
        MergePass(TR, L->r, k, L->length);
        k = 2*k;
    }
}

快速排序

快速排序(Quick Sort)的基本思想: 通过⼀趟排序将待排序记录分割成独⽴的两部分; 其中⼀部分记录的关键字均为另⼀部分记录的关键字⼩,则可分别对两部分记录继续进⾏排序,以达到整个排序有序的⽬的。

代码

int Partition(SqList *L, int low, int high) {
    //默认中枢轴是第一个元素
    int pivotkey = L->r[low];
    
    while (low < high) {
        //从后往前获取到小于中枢轴值的元素,找到后与中枢轴交换
        while (low < high && L->r[high] >= pivotkey) {
            high--;
        }
        swap(L, low, high);
        //从前往后获取到大于中枢抽值的元素,找到后与中枢轴交换
        while (low < high && L->r[low] <= pivotkey) {
            low++;
        }
        swap(L, low, high);
    }
    
    return low;
}

void QSort(SqList *L, int low, int high) {
    if (low < high) {//递归出口
        int pivot = Partition(L, low, high);//获取中枢轴
        printf("pivot = %d L->r[%d] = %d \n", pivot, pivot, L->r[pivot]);
        //递归处理,分两部分,[low,pivot-1],[pivot+1,high]
        QSort(L, low, pivot-1);
        QSort(L, pivot + 1, high);
    }
}

void QucikSort(SqList *L) {
    QSort(L, 1, L->length);
}

代码实现逻辑:

  1. 默认第一个元素为中枢轴,把小于它的的元素放到前面,把大于它的元素放到后面。
  2. 根据中枢值,将原数组分成两部分
  3. 新分成的两部分记录递归第一步。
  4. 当low不小于high时,为递归出口。

优化快速排序

优化的点:

  1. 如果默认数组中low对应的值本来就是最小值,或者数组中部分元素是有序的,例如[2,3,4,5,6,7,9,8]这种情况每次再找中枢值的时候,进行的比较次数都很多。此处优化方案是使用low、high和mid值进行比较,取中间值的方式。
  2. 用哨兵记录中枢值,这样在while循环中,不调用交换方法,而是用中间值的方式进行赋值。

代码:

int Partition2(SqList *L, int low, int high) {
    //优化获取中枢轴,如果下标low对应的值是最小值 或者 high对应的值是最大值,这要快排的效率会很低
    //所以我们获取low、high、和中间下标m对应的三个值,进行比较,把中间值放在low的位置
    int m = (low + high) / 2;
    //low和high比较,相对小的放在low上
    if (L->r[low] > L->r[high]) {
        swap(L, low, high);
    }
    //m和high比较,把相对小的放在m上
    if (L->r[m] > L->r[high]) {
        swap(L, m, high);
    }
    //m和low比较,把相对大的放low上
    if (L->r[m] > L->r[low]) {
        swap(L, m, low);
    }
    
    //获取中枢值
    int pivotkey = L->r[low];
    //用哨兵记录中枢值,这样就不用频繁的进行交换了
    L->r[0] = pivotkey;
    
    while (low < high) {
        while (low < high && L->r[high] >= pivotkey) {
            high--;
        }
        //没有进行交换,只是赋值操作
        L->r[low] = L->r[high];
        
        while (low < high && L->r[low] <= pivotkey) {
            low++;
        }
        //没有进行交换,只是赋值操作
        L->r[high] = L->r[low];
    }
    //将哨兵记录的值赋值给下标low上
    L->r[low] = L->r[0];
    return low;
}

#define MAX_LENGTH_INSERT_SORT 7 //数组长度的阀值
void QSort2(SqList *L, int low, int high) {
    if ((high - low) > MAX_LENGTH_INSERT_SORT) {
        int pivot = Partition(L, low, high);
        printf("pivot = %d L->r[%d] = %d\n",pivot,pivot,L->r[pivot]);
        QSort(L, low, pivot-1);
        QSort(L, pivot+1, high);
    } else {
        InsertSort(L);
    }
}

void QuickSort2(SqList *L) {
    QSort2(L, 1, L->length);
}

总结