排序

插入类排序

直接插入排序:每趟将一个待排序的关键字按照其值的大小插入到已经排好的部分有序序列的适当位置上

排序过程
元数据: 49 38 65 97 76 13 27 49
第一次排序:49 38 65 97 76 13 27 49
第二次排序:38 49 65 97 76 13 27 49
第三次排序:38 49 65 97 76 13 27 49
第四次排序:38 49 65 76 97 13 27 49
第五次排序:13 38 49 65 97 76 27 49
第六次排序:13 38 49 65 76 97 27 49
第七次排序:13 38 27 49 65 97 76 49
第八次排序:13 38 27 49 49 65 97 76

代码实现

void insertSort(int R[],int n){
    int i,j;
    int temp;
    for (i=1; i<n; i++) {
        temp = R[i];
        j = i-1;
        while (j>=0&&temp<R[j]) {
            R[j+1] = R[j];
            j--;
        }
        R[j+1] = temp;
    }
}

折半插入排序: 折半插入排序的算法和直接插入排序的算法思想类似,区别是查找插入位置的方法不同,折半插入排序是采用折半查找法来查找插入位置

元数据:13 38 49 65 76 97(有序部分),代插入的数据:27 49
27插入流程:
low=0,high=5,m=[(0+5)/2],下标为2的关键字是49,所以应该插入到49的低半区;high=m-1,low=0
low=0,high=1,m=[(0+1)/2]=0,下标为0的关键字是13,所以应该插入到13的高半区;low=m+1,high=1
low=1,high=1,m=[(1+1)/2]=1,下标为0的关键字是38,所以应该插入到38的低半区,改变high=mid-1=0,high<low,折半查找结束,27放到high之后,即13之后,此时的序列为:
13 27 38 49 65 76 97
依照此方法插入剩余结点

希尔排序:又叫做缩小增量排序,其本质还是插入排序,只不过是将待排序列按照某种规则分成几个子序列,分别对这几个子序列进行直接插入排序

元数据:49 38 65 97 76 13 27 49 55 04
1)以5为增量分割序列,得到以下几个子序列
                49                        13
                      38                       27
                           65                       49
                                 97                      55
                                       76                      04
分别对5个子序列进行直接插入排序
                13                        49
                      27                      38
                           49                       65
                                 55                      97
                                       04                      76
一趟希尔排序结束,结果为:
                 13 27 49 55 04 49 38 65 97 76
2)再对上面的结果以增量3分割,得到以下几个子序列
                 13             55            38            76
                      27             04             65
                            49             49             97
分别对三个子序列进行直接插入排序
                 13             38            55            76
                      04             27             65
                            49             49             97
第二趟希尔排序结束,结果为:
                 13 04 49 38 27 49 55 65 97 76
3)最后以增量为1进行分割,即对以上结果全体的关键字进行一趟直接插入排序,结果为:
                 04 13 27 38 49 49 55 65 76 97

交换类排序

冒泡排序:它重复地走访过要排序的元素列,依次比较两个相邻的元素,如果他们的顺序(如从大到小、首字母从A到Z)错误就把他们交换过来。走访元素的工作是重复地进行直到没有相邻元素需要交换,也就是说该元素已经排序完成。

原始序列: 49 38 65 97 76 13 27 49
一趟冒泡排序: 38 49 65 76 12 27 49 97
将最大的元素97移到了最后,之后对前n-1位进行排序:
两趟冒泡排序: 38 49 65 12 27 49 76 97
之后依次类推知道得到有序序列
三趟冒泡排序: 38 49 12 27 49 65 76 97
四趟冒泡排序: 38 12 27 49 49 65 76 97
五趟冒泡排序: 12 24 38 49 49 65 76 97

代码实现

void bubbleSort(int R[],int n){
    bool flag = false;
    int i,j;
    for (i=n-1; i>=1; i--) {
        flag = false;
        for (j = 1; j<=i; j++) {
            if (R[j-1]>R[j]) {
                int temp = R[j-1];
                R[j-1] = R[j];
                R[j] = temp;
                flag = true;
            }
        }
        if (!flag) {
            return;
        }
    }
}

快速排序:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。

timg-2.jpeg

代码实现

void quickSort(int R[],int low,int high){
    int temp;
    int i = low,j=high;
    if (i<j) {
        temp=R[low];
        while (i!=j) {
            while (j>i && R[j]>temp) {
                --j;
            }
            if (i<j) {
                R[i] = R[j];
                ++i;
            }
            
            while (j>i && R[i]<temp) {
                ++i;
            }
            if (i<j) {
                R[j] = R[i];
                --j;
            }
        }
        R[i] = temp;
        quickSort(R, low, i-1);
        quickSort(R, i+1, high);
    }
}

选择类排序

简单选择排序:从头至尾顺序扫描序列,找到最小的一个关键字和第一个关键字交换,接着从剩下的关键字中继续这种选择和交换,最终使序列有序

image.png

代码实现

void selectSort(int R[],int n){
    int i,j,k;
    for (i=0; i<n; i++) {
        k = i;
        for (j=i+1; j<n; j++) {
            if (R[j]>R[k]) {
                k = j;
            }
        }
        int temp = R[i];
        R[i] = R[k];
        R[k] = temp;
    }
}

堆排序:是指利用堆的一种排序算法。堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。
堆可以分为大顶堆和小顶堆,若父亲大孩子小则叫做大顶堆,否则叫做小顶堆
堆的构造

image.png

插入节点

需要在插入节点后保持堆的性质,需要先将插入的结点x放到最底层的右边,满足完全二叉树的特点,然后把x依次向上调整到合适的位置以满足堆的性质

删除结点

删除结点时,原来的位置就会出现一个孔,将最底层最右面的叶子值赋值给该孔并下调到合适的位置,最后把叶子删除

排序

以上调整已经建立的一个大顶堆,对应的序列为:97 76 65 49 49 13 27 38,将97和38交换,将97移除放入有序序列中,此时把除97外的序列进行排序


image.png

依次类推,知道全部都排好序

归并类排序

二路归并排序:是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。
排序过程

原始数据:49 38 65 97 76 13 27

1)将原始序列看成7个只含有一个关键字的子序列,显然这些子序列都是有序的
2)两两归并,形成若干有序元组,即49 38 归并成{38 49},第一趟归并结束后为
{38 49} {65 97} {13 76} {27}
3)继续两两归并形成四元组
{38 49 65 97} {13 27 76}
4)再进行一次归并
13 27 38 49 65 76 97

基数类排序

属于“分配式排序”(distribution sort),又称“桶子法”(bucket sort)或bin sort,基数排序的思想是“多关键字排序”,顾名思义,它是透过键值的部份资讯,将要排序的元素分配至某些“桶”中,藉以达到排序的作用,基数排序法是属于稳定性的排序,其时间复杂度为O (nlog(r)m),其中r为所采取的基数,而m为堆数,在某些时候,基数排序法的效率高于其它的稳定性排序法。
排序过程

原数据:278 109 063 930 589 184 505 269 008 083


image.png

以上为九种排序算法的介绍,接下来总结下算法的时间复杂度,空间复杂度,稳定性以及使用场景

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

推荐阅读更多精彩内容

  • 一、概述 排序算法概念 在计算机科学与数学中,一个排序算法是将一组杂乱无章的数据按一定的规律顺次排列起来的算法。排...
    简书冷雨阅读 1,030评论 0 0
  • 1、常用排序算法 2、快速排序法 基本思想:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比...
    Bling_ll阅读 541评论 0 0
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 5,178评论 0 52
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 1,250评论 0 2
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    闲云清烟阅读 757评论 0 6