排序

冒泡排序(Bubble Sort) ⼀种交换排序,它的基本思想就是: 两两⽐较相邻的记录的关键字,如果反序则交换,直到没有反序的记录为⽌

两两比较,往上移动 -> 第二层遍历是从 尾部开始的

遍历中, 没有任何数据的交换动作 -> 数据是有序的

void BubbleSort(SqList *L){

    int i,j;

    //flag用作标记

    Status flag =TRUE;

    //I 从[1,L->length) 遍历;

    //如果flag为False退出循环. 表示已经出现过一次j从L->Length-1 到 i的过程,都没有交换的状态;

    for(i =1; i < L->length&& flag; i++) {

        //flag 每次都初始化为FALSE

        flag =FALSE;

        for(j = L->length-1; j>=i; j--) {

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

            //交换L->r[j]和L->r[j+1]值;

            swap(L, j, j+1);

            //如果有任何数据的交换动作,则将flag改为true;

            flag = TRUE;

            }

        }

        if (!flag) break;

    }

}

直接插⼊排序算法(Stight Inserton Sort)的基本操作是将⼀个记录插⼊到已经排好序的有序表中,从⽽得到⼀个新的,记录数增1的有序表

从第二个元素开始,往已经排好序的数据中插入

void InsertSort(SqList *L){

    int i,j;

    int temp=0;

    //假设排序的序列集是{0,5,4,3,6,2};

    //i从2开始的意思是我们假设5已经放好了. 后面的牌(4,3,6,2)是插入到它的左侧或者右侧

    for(i=2;i<=L->length;i++)

    {

        //需将L->r[i]插入有序子表

        if(L->r[i]r[i-1])

        {

            //设置哨兵 可以把temp改为L->r[0]

            temp = L->r[i];

            for(j=i-1;L->r[j]>temp;j--)

                    //记录后移

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

            //插入到正确位置 可以把temp改为L->r[0]

            L->r[j+1]=temp;

        }

    }

}

简单排序算法(Simple Selecton Sort) 就是通过n-i次关键词⽐较,从n-i+1个记录中找出关键

字最⼩的记录,并和第i(1<=i<=n) 个记录进⾏交换

找出最小值,放在第一位,依此类推...

void SelectSort(SqList *L){

    int i,j,min;

    for(i =1; i < L->length; i++) {

        //① 将当前下标假设为最小值的下标

        min = i;

        //② 循环比较i之后的所有数据

        for(j = i+1; j <= L->length; j++) {

            //③ 如果有小于当前最小值的关键字,将此关键字的下标赋值给min

            if(L->r[min] > L->r[j]) {

                min = j;

            }

        }

        //④ 如果min不等于i,说明找到了最小值,则交换2个位置下的关键字

        if(i!=min)

            swap(L, i, min);

    }

}

堆排序

堆结构:堆是一个的完全⼆叉树。具备完全⼆叉树的特点,左结点,右结点,父节点,树结点个数的 特点

 每个结点的值都⼤于或等于其左右孩⼦结点的值,称为⼤顶堆

每个结点的值都小于或等于其左右孩⼦结点的值,称为小顶堆

堆排序思路:

1. 将数据按照层序遍历形成一个完全二叉树, 按照⼤顶堆或⼩顶堆需求,对完全二叉树进行调整,调整的是非叶子结点

2. 将堆顶元素与末尾元素交换,将最⼤元素”沉"到数组末端; 

3. 重新调整结构,使其满⾜堆定义,然后继续交换堆顶元素与当前末尾元素,反复执⾏调整+交换步骤,直到整个序列有序

大顶堆调整函数

 条件: 在L.r[s...m] 记录中除了下标s对应的关键字L.r[s]不符合大顶堆定义,其他均满足;

 结果: 调整L.r[s]的关键字,使得L->r[s...m]这个范围内符合大顶堆定义.

void HeapAjust(SqList *L,int s,int m){

    int temp,j;

    //① 将L->r[s] 存储到temp ,方便后面的交换过程;

    temp = L->r[s];

    //② j 为什么从2*s 开始进行循环,以及它的递增条件为什么是j*2

    //因为这是颗完全二叉树,而s也是非叶子根结点. 所以它的左孩子一定是2*s,而右孩子则是2s+1;(二叉树性质5)

    for(j =2* s; j <=m; j*=2) {

        //③ 判断j是否是最后一个结点, 并且找到左右孩子中最大的结点;

        //如果左孩子小于右孩子,那么j++; 否则不自增1. 因为它本身就比右孩子大;

        if(j < m && L->r[j] < L->r[j+1])

            ++j;

        //④ 比较当前的temp 是不是比较左右孩子大; 如果大则表示我们已经构建成大顶堆了;

        if(temp >= L->r[j])

            break;

        //⑤ 将L->[j] 的值赋值给非叶子根结点

        L->r[s] = L->r[j];

        //⑥ 将s指向j; 因为此时L.r[4] = 60, L.r[8]=60. 那我们需要记录这8的索引信息.等退出循环时,能够把temp值30 覆盖到L.r[8] = 30. 这样才实现了30与60的交换;

        s = j;

    }

    //⑦ 将L->r[s] = temp. 其实就是把L.r[8] = L.r[4] 进行交换;

    L->r[s] = temp;

}

堆排序--对顺序表进行堆排序

void HeapSort(SqList *L){

    int i;

    //1.将现在待排序的序列构建成一个大顶堆;

    //将L构建成一个大顶堆;

    //i为什么是从length/2.因为在对大顶堆的调整其实是对非叶子的根结点调整.

    for(i=L->length/2; i>0;i--){

        HeapAjust(L, i, L->length);

    }

    //2.逐步将每个最大的值根结点与末尾元素进行交换,并且再调整成大顶堆

    for(i = L->length; i >1; i--){

        //① 将堆顶记录与当前未经排序子序列的最后一个记录进行交换;

        swap(L,1, i);

        //② 将L->r[1...i-1]重新调整成大顶堆;

        HeapAjust(L,1, i-1);

    }

}

堆排序的时间复杂度为:O(nlogn)

堆排序是就地排序,空间复杂度为常数:O(1)

归并排序

合并

//③ 将有序的SR[i..mid]和SR[mid+1..n]归并为有序的TR[i..n]

void Merge(intSR[],intTR[],inti,intm,intn)

{

    int j,k,l;

    //1.将SR中记录由小到大地并入TR

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

    {

        if (SR[i]<SR[j])

            TR[k]=SR[i++];

        else

            TR[k]=SR[j++];

    }


    //2.将剩余的SR[i..mid]复制到TR

    if(i<=m)

    {

        for(l=0;l<=m-i;l++)

            TR[k+l]=SR[i+l];

    }

    //3.将剩余的SR[j..mid]复制到TR

    if(j<=n)

    {

        for(l=0;l<=n-j;l++)

            TR[k+l]=SR[j+l];

    }

}

1.递归实现: 将数据 拆成长度1的子序列, 再将子序列两两进行合并且排序

先拆分,后合并

//② 将SR[s...t] 归并排序为 TR1[s...t];

void MSort(int SR[],int TR1[],int low,int hight){

    int mid;

    int TR2[MAXSIZE+1];

    if(low == hight)

        TR1[low] = SR[low];

    else{

        //1.将SR[low...hight] 平分成 SR[low...mid] 和 SR[mid+1,hight];

        mid = (low + hight)/2;

        //2. 递归将SR[low,mid]归并为有序的TR2[low,mid];

        MSort(SR, TR2, low, mid);

        //3. 递归将SR[mid+1,hight]归并为有序的TR2[mid+1,hight];

        MSort(SR, TR2, mid+1, hight);

        //4. 将TR2[low,mid] 与 TR2[mid+1,hight], 归并到TR1[low,hight]中

        Merge(TR2, TR1, low, mid, hight);

    }

}

//① 对顺序表L进行归并排序

void MergeSort(SqList *L){

    MSort(L->r,L->r,1,L->length);

}

2.非递归实现:

边拆分,边合并

边拆分边合并

//对SR数组中相邻长度为s的子序列进行两两归并到TR[]数组中;

void MergePass(int SR[],int TR[],int s,int length){

    int i =1;

    int j;

    //①合并数组

    //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

    // 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(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;

    }

}

排序总结

排序总结

稳定性解释:排序后的元素相同元素的顺序依然是排序之前的顺序

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

推荐阅读更多精彩内容