[从零开使学算法]排序算法Ⅱ

书接上回,我们继续介绍排序算法。
笔者的编程环境:

  1. 语言:c++
  2. 编译器: Code Blocks
  3. 系统环境:window 10 x64

快速排序

基本思想

  1. 在待排序数组中任选一个元素作为枢轴pivot(也有叫基准base的)
  2. 一趟排序使得元素组划分成两部分,一部分元素均大于枢轴,一部分均小于枢轴。
  3. 对划分后的两部分继续执行第一步与第二部,至序列有序

空间复杂度:O(logn)
时间复杂度:O(nlogn)
稳定性:不稳定
代码如下:

template<typename T>
int partition(T arr[],int low,int high)//划分
{
    T index=arr[low];
    while(low<high){
        while(low<high&&arr[high]>=index)high--;
        arr[low]=arr[high];
        while(low<high&&arr[low]<=index)low++;
        arr[high]=arr[low];
    }
    arr[low]=index;
    return low;
}
template<typename T>
int quickSort(T arr[],int low,int high)//排序
{
    if(low<high){
        int index=partition(arr,low,high);
        quickSort(arr,low,index-1);
        quickSort(arr,index+1,high);
    }
    return 0;
}

template<typename T>
int sort(T arr[],int count)//排序入口
{
    quickSort(arr,0,count-1);
    return 0;
}

堆排序

基本思想

  1. 将待排序序列构建成堆(以大根堆为例)
  2. 将对顶元素与堆中最后一个元素交换,并将最后一个元素脱离堆
  3. 对剩下的元素进行堆调整,使其依然是大根堆。
  4. 重复执行2,3步至堆消失

空间复杂度:O(1)
时间复杂度:O(nlogn)
稳定性:不稳定
代码如下:

template<typename T>
int swap(T *a,T *b)//交换函数
{
    T c;
    c=*a;
    *a=*b;
    *b=c;
    return 0;

}

template<typename T>
int buildHeap(T arr[],int count)//建堆函数
{
    for(int i=count/2-1;i>=0;i--)
    {
        heapAdjust(arr,count,i);
    }
    return 0;
}

template<typename T>
int heapAdjust(T arr[],int count,int begin)//堆调整函数
{
    T index=arr[begin];
    for(int i=begin*2;i<count;i*=2)
    {
        if(i<count-1&&arr[i]<arr[i+1])i++;
        if(index>=arr[i])break;
        else{
            arr[begin]=arr[i];
            begin=i;
        }
    }
    arr[begin]=index;
    return 0;
}

template<typename T>
int sort(T arr[],int count)//堆排序
{
    buildHeap(arr,count);
    for(int i=count-1;i>0;i--)
    {
        swap(arr[i],arr[0]);
        heapAdjust(arr,i,0);//通过不断调整堆的大小使最后一个元素脱离堆
    }
    return 0;
}

归并排序(二路归并)

基本思想

  1. 将待排序序列划分成n个组
  2. 将得到的组两两合并得到一定数量的有序组
  3. 重复执行第二步至仅剩一个有序组

示意图如下:
在这里插入图片描述

空间复杂度:O(n)
时间复杂度:O(nlogn)
稳定性:稳定
代码如下:

//需要提前准备一个与待排序数组同类型同长度的数组temp
//最终结果在原待排序数组中
template<typename T>
int sort(T arr[],int count,T temp[])//排序入口
{
    mergeSort(arr,0,count-1,temp);
    return 0;
}

template<typename T>
int mergeSort(T arr[],int low,int high,T temp[])//划分
{
    if(low<high)
    {
        int mid=(low+high)/2;
        mergeSort(arr,low,mid,temp);
        mergeSort(arr,mid+1,high,temp);
        merge(arr,low,mid,high,temp);
    }
    return 0;
}

template<typename T>
int merge(T arr[],int low,int mid,int high,T temp[])//合并
{
    for(int k=low; k<=high; k++)temp[k]=arr[k];
    int i=low,j=mid+1,k=low;
    for(; i<=mid&&j<=high; k++)arr[k]=temp[temp[i]<=temp[j]?i++:j++];
    while(i<=mid)arr[k++]=temp[i++];
    while(j<=high)arr[k++]=temp[j++];
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值