排序算法

十大排序算法总结


基础交换函数

void swap(int& temp1, int& temp2)
{
    temp1 = temp1 ^ temp2;
    temp2 = temp1 ^ temp2;
    temp1 = temp1 ^ temp2;
}

冒泡排序

算法流程
  • 比较相邻元素,如果满足条件则交换(视从小到大排,和从大到小排定)
  • 顺序比较,直到最后一位,则最后一位已排好序
  • 重复以上步骤,但已经排好的不参与比较。
复杂度分析
  • 时间复杂度(平均): O(n^2)
  • 时间复杂度(最好): O(n) 初始已排序(改进算法才可以)
  • 时间复杂度(最坏): O(n^2)
  • 是否稳定: 稳定
C代码

冒泡排序

void bubbleSort(int array[], int length) 
{

    for (int i = 0; i<length-1; i++)  // 这一步控制步长
    {
        for (int j = 0; j < length -1 -i;j++) // 这一步前后排序
        {
            if (array[j] > array[j+1])
            {
                swap(array[j], array[i]);
            }
        }
    }
}

选择排序

算法流程
  • 遍历比较确定数组中最小值所在的下标
  • 最小值与第一位未排序好的数交换
  • 重复以上步骤,但已经排好的不参与比较。
复杂度分析
  • 时间复杂度(平均): O(n^2)
  • 时间复杂度(最好): O(n) 初始已排序(改进算法才可以)
  • 时间复杂度(最坏): O(n^2)
  • 是否稳定: 稳定
C代码

选择排序

void selectSort(int array[], int length) {
    int minIndex;
    for (int i = 0; i < length - 1; i++) {
        minIndex = i;
        for (int j = i + 1; j < length; j++) {
            if ( array[j] < array[minIndex] ) {     // 寻找最小的数
                minIndex = j;                       // 将最小数的索引保存
            }
        }
        swap(array[i], array[minIndex]);
    }
}

插入排序

算法流程
  • 类似斗地主叠排
  • 默认第一个已经排好,待插入的值与前面的比较,若小于,则前后的后移
  • 直到待插入的值大于前一个值
  • 未排序的数重复以上步骤。
复杂度分析
  • 时间复杂度(平均): O(n^2)
  • 时间复杂度(最好): O(n)
  • 时间复杂度(最坏): O(n^2)
  • 是否稳定: 稳定
C代码

插入排序

void insertSort(int array[], int length)
{
    int preIndex, current;
    for (int i = 1; i < length; i++) {
        preIndex = i - 1;
        current = array[i];
        while (preIndex >= 0 && array[preIndex] > current) {
            array[preIndex + 1] = array[preIndex];
            preIndex--;
        }
        array[preIndex + 1] = current;
    }
}

希尔排序

算法流程
  • 确定间隔数组,这里采用3的倍数递增法,可换,最后一个间隔需为1
  • 按照不同的间隔进行直接插入排序
复杂度分析
  • 时间复杂度(平均): O(n^1.3) 统计结果
  • 时间复杂度(最好): O(n)
  • 时间复杂度(最坏): O(n^2)
  • 是否稳定: 不稳定 间隔排序导致
C代码

希尔排序

void  shellSort(int array[], int length)
{
    int gap = 1, current;
    int i, preIndex;
    while (gap < length / 3) {          // 动态定义间隔序列
        gap = gap * 3 + 1;              // 保证除3的时候会取到1
    }
    for ( ; gap > 0; gap = floor(gap / 3 ) ) {
        // 间隔为gap的直接插入排序,这里除了前几个数外,都需要与之前的比较
        for (i = gap; i < length; i++) {
            current = array[i];
            preIndex = i - gap;
            // 注意取值范围
            while (preIndex >= 0 && array[preIndex] > current) {
                array[preIndex + gap] = array[preIndex];
                preIndex = preIndex - gap;
            }
            array[preIndex + gap] = current;
        }
    }
    
}

归并排序

算法流程
  • 参考算法导论的分而治之的思想
  • 递归将大的数组分为左右两部分,一直分到只剩最后一个值
  • 逆递归merge排序,开辟新空间将两个数组最小值比较,排序到原数组空间中
  • 技巧是设置无穷大的哨兵值,毕竟不断的判断是否数组已经遍历到结尾
复杂度分析
  • 时间复杂度(平均): O(nlog(n))
  • 时间复杂度(最好): O(nlog(n))
  • 时间复杂度(最坏): O(nlog(n))
  • 是否稳定: 稳定
C代码

合并函数

void merge(int array[], int begin, int middle, int end) {
    int len1 = middle - begin + 1;
    int len2 = end - middle;
    int i, j;
    // 定义中间变量
    int *arr1 = new int[len1 + 1];
    int *arr2 = new int[len2 + 1];

    // 赋初值
    for (i = 0; i < len1; i++)
    {
        *(arr1 + i) = array[begin + i];
    }
    for (j = 0; j < len2; j++)
    {
        *(arr2 + j) = array[middle + j + 1];
    }
    // 设置哨兵
    arr1[len1] = 10000;
    arr2[len2] = 10000;

    // 从最前面遍历
    i = 0;
    j = 0;
    for (int k = begin; k <= end; k++)
    {
        if (arr1[i] < arr2[j])
        {
            array[k] = arr1[i];
            i++;
        }
        else
        {
            array[k] = arr2[j];
            j++;
        }
    }

    delete arr1;
    delete arr2;
}

归并排序

void mergeSort(int array[], int begin, int end)
{
    int middle = floor((end + begin) / 2);
    // 先分后和
    if (begin < end) {
        mergeSort(array, begin, middle);
        mergeSort(array, middle + 1, end);
        merge(array, begin, middle, end);
    }

}

快速排序

算法流程
  • 选择第一个为基数,将余下的数按照是否大于或者小于,归于两边
  • 为了使用原空间,采用左右震荡比较方式比较
  • 最小值与第一位未排序好的数交换
  • 分左右递归以上步骤。
复杂度分析
  • 时间复杂度(平均): O(nlog(n))
  • 时间复杂度(最好): O(n^2) 初始无序
  • 时间复杂度(最坏): O(nlog(n))
  • 是否稳定: 不稳定 左右震荡导致
C代码

快速排序

void quickSort(int array[], int left, int right)
{

    if (left < right) {
        int temp = array[left];
        int indexL = left;
        int indexR = right;
        while (indexR > indexL)
        {
            // 只要有涉及下标运算,都需要判断
            // 右振荡
            while (indexR > indexL && array[indexR] >= temp)
            {
                indexR--;
            }
            if(indexR > indexL)
                array[indexL++] = array[indexR];

            // 左振荡
            while (indexR > indexL  && array[indexL] < temp)
            {
                indexL++;
            }
            if (indexR > indexL)
                array[indexR--] = array[indexL];
        }

        // 计算完后 indexR = indexL = 最中间的下标 
        array[indexR] = temp;

        // 递归
        quickSort(array, left, indexR - 1);
        quickSort(array, indexR + 1, right);
    }

}

堆排序

算法流程
  • 主要过程分为调整堆,将为排好序的最大值交换到已排好序的最后
  • 重复以上步骤,但已经排好的不参与比较。
  • 这里一个技巧就是利用 二叉堆,根节点与子节点的关系。
复杂度分析
  • 时间复杂度(平均): O(nlog(n))
  • 时间复杂度(最好): O(nlog(n))
  • 时间复杂度(最坏): O(nlog(n))
  • 是否稳定: 不稳定 调整导致
C代码

堆排序

void adjustHeap(int array[], int len)
{
    int maxIndex, rootIndex, subLeftIndex, subRightIndex;
    // 从下往上调整
    for (int i = len / 2; i > 0; i--){
        // C的下标从0开始
        rootIndex = i - 1;
        maxIndex = rootIndex;
        subLeftIndex = 2 * i -1;  
        subRightIndex = 2 * i;
        if (array[maxIndex] < array[subLeftIndex])
            maxIndex = subLeftIndex;
        // 右子节点需要判断是否越界
        if ( subRightIndex< len && array[maxIndex] < array[subRightIndex])
            maxIndex = subRightIndex;
        // 根节点与最大节点交换
        if( maxIndex != rootIndex)
            swap(array[rootIndex], array[maxIndex]);
    }
}

void heapSort(int array[], int len)
{
    int adjustLen = len;
    for (int i = 0; i < len - 1 ; i++) {
        adjustHeap(array, adjustLen);
        // C的下标从0开始的
        swap(array[0], array[adjustLen-1]);
        adjustLen--;
    }

}

计数排序

算法流程
  • 条件限制: 待排序数组一定是一定范围内的整数
  • 首先取得最大最小值,确定辅助数组大小k。
  • 遍历待排序数组,与值对应,辅助数组相应位置+1。遍历后辅助数组相应下标存放值的数目
  • 遍历将值按顺序存回原数组,则排序完成
复杂度分析
  • 时间复杂度(平均): O(n+k)
  • 时间复杂度(最好): O(n+k) 初始已排序(改进算法才可以)
  • 时间复杂度(最坏): O(n+k)
  • 是否稳定: 稳定
C代码

计数排序

void countSort(int array[], int len)
{

    int minValue = array[0];
    int maxValue = array[0];
    // 获得最小最大值,确定排序边界
    for (int i = 1; i < len; i++) {
        if (array[i] < minValue)
            minValue = array[i];
        if (array[i] > maxValue)
            maxValue = array[i];
    }
    
    int countLen = maxValue - minValue + 1;
    int * countArray = new int[countLen];
    // 清0
    for (int i = 0; i < countLen; i++) {
        countArray[i] = 0;
    }
    // 计算数目
    for (int i = 0; i < len; i++) { 
        countArray[array[i] - minValue] ++;
    }

    // 计算数目
    int j = 0;
    for (int i = 0; i < countLen; i++) {
        while ( countArray[i] != 0 )
        {
            array[j++] = i + minValue;
            countArray[i]--;
        }
    }

    delete countArray;
}

桶排序

算法流程
  • 基于分组排序思想
  • 设置一个定量的数组当作空桶
  • 遍历输入数据,并且把数据一个一个放到对应的桶里去
  • 对每个不是空的桶进行排序;
  • 把排好序的数据拼接起来
复杂度分析
  • 时间复杂度(平均): O(n+k)
  • 时间复杂度(最好): O(n)
  • 时间复杂度(最坏): O(n^2)
  • 是否稳定: 稳定
C代码

基数排序

算法流程
  • 取得数组中的最大数,并取得位数
  • 从最低位开始取每个位组成 待排数组
  • 待排数组进行计数排序
复杂度分析
  • 时间复杂度(平均): O(n*k)
  • 时间复杂度(最好): O(n*k)
  • 时间复杂度(最坏): O(n*k)
  • 是否稳定: 稳定
C代码
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容

  • 概述 因为健忘,加上对各种排序算法理解不深刻,过段时间面对排序就蒙了。所以决定对我们常见的这几种排序算法进行统一总...
    清风之心阅读 695评论 0 1
  • 排序算法基础 排序算法,是一种能将一串数据按照特定的排序方式进行排列的一种算法,一个排序算法的好坏,主要从时间复杂...
    jackyshan阅读 3,941评论 3 11
  • 本文分析冒泡、选择、插入、希尔、快速、归并和堆排序,为不影响阅读体验,将关于时间、空间复杂度和稳定性的概念放在博文...
    DeppWang阅读 427评论 0 2
  • 所有的美好都值得被纪念。日子一定要仔仔细细的过,认真在意的过,把每一个细枝未节,每一分钟,每一秒钟,都要仔细玩味...
    柠檬听你说阅读 1,616评论 0 0
  • 腾讯新闻,1月28日新闻的标题是:除夕夜 全球的寺庙都被挤爆了 咱们中国人有13亿人口,又是出了名爱扎堆凑热闹,...
    昕城阅读 155评论 0 0