排序问题

排序问题

<!--more-->

排序方法        平均情况        最好情况        最坏情况        辅助空间        稳定性

冒泡排序        O(n^2)          O(n)              O(n^2)            O(1)                稳定

选择排序        O(n^2)          O(n^2)            O(n^2)            O(1)              不稳定

插入排序        O(n^2)          O(n)              O(n^2)            O(1)                稳定

希尔排序O(n*log(n))~O(n^2) O(n^1.3)      O(n^2)            O(1)              不稳定

堆排序          O(nlog(n))    O(nlog(n))    O(n*log(n))      O(1)              不稳定

归并排序      O(nlog(n))    O(nlog(n))    O(n*log(n))      O(n)                稳定

快速排序      O(nlog(n))    O(nlog(n))      O(n^2)            O(1)              不稳定

关于稳定性: 假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次序保持不变,即在原序列中,ri=rj,且ri在rj之前,而在排序后的序列中,ri仍在rj之前,则称这种排序算法是稳定的;否则称为不稳定的。

(一)冒泡排序

基本思想 :在一组没有排序的数组中,通过自上而下对相邻的两个数进行比较,让较大的数下沉

即:每当两相邻的数比较后发现它们的排序与排序要求相反时,就将它们互换

时间复杂度 :无论给定什么数列,都需要比较 n(n-1),则为 O(n^2)

// 冒泡排序,返回升序数组

    publicstaticvoidbubbleSort(inta[],intn) {

        for(inti=0;i<n-1;i++) {

            for(intj=0;j<n-1-i;j++)

                if(a[j]>a[j+1]) {

                    inttmp=a[j];

                    a[j]=a[j+1];

                    a[j+1]=tmp;

            }

        }

    }

改进的冒泡排序 :

改进后,当传入的数组为已经排序的的时,只需进行 n 次比较,则为  O(n),为最优情况

对冒泡排序常见的改进方法是加入一标志性变量flag,用于标志某一趟排序过程中是否有数据交换,如果进行某一趟排序时并没有进行数据交换,则说明数据已经按要求排列好,可立即结束排序,避免不必要的比较过程

// 冒泡排序,返回升序数组

    publicstaticvoidbubbleSort_1(inta[],intn) {

        booleanflag;

        for(inti=0;i<n-1;i++) {

            flag=false;

            for(intj=0;j<n-1-i;j++)

                if(a[j]>a[j+1]) {

                    inttmp=a[j];

                    a[j]=a[j+1];

                    a[j+1]=tmp;

                    flag=true;

                }

            if(flag==false)

                break;

        }

    }

对于每次的排序,记录下交换的位置pos,则当一次排序结束后,可以得到最后一次排序的坐标,当下一趟排序开始时,只需要比较到pos 就可以了,因为pos 后面的都已经排序好了。

publicstaticvoidbubbleSort_2(intr[],intn) {

    inti=n-1;// 初始时,最后位置保持不变

    while(i>0) {

        intpos=0;// 每趟开始时,无记录交换

        for(intj=0;j<i;j++)

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

                pos=j;// 记录交换的位置

                inttmp=r[j];

                r[j]=r[j+1];

                r[j+1]=tmp;

            }

        i=pos;// 为下一趟排序作准备

    }

}

(二)选择排序

算法思想 :在第一次排序,扫描N个数据,选出其中的最小值,与第一个元素交换,接着进行第二趟,以此类推

复杂度 : 平均时间复杂度:O(n2)。当有重复元素时,会改变位置,比如【2,2,5】,第一趟第1,2就会交换位置,所以此算法为不稳定的。

    // 选择排序

    staticvoidselectionSort(inta[],intn) {

        intindex;

        for(inti=0;i<n;i++) {

            index=i;

            for(intj=i+1;j<n-1;j++) {

                if(a[j]<a[index])

                    index=j;

                if(index!=i) {

                    inttmp=a[index];

                    a[index]=a[i];

                    a[i]=tmp;

                }

            }

        }

    }

(三) 插入排序

算法思想 :把数据分成有序组和插入组,一般把第一个元素当成有序组,然后从插入组中拿第一个数据,从有序组的最后一个元素进行比较,找到合适的位置并插入,为稳定排序。

时间复杂度 :当给定的数据为排序升序时,只要比较N 次,为O(n);当是降序时,要 n(n+1)/2,所以为O(n^2)

平均的时间复杂度为O(n^2)。

//插入排序

    publicstaticvoidInsertionSort(inta[],intn)

    {

    inti=0;

    intj=0;

    inttmp=0;

    for(i=1;i<n;i++)

    {

    tmp=a[i];//从待插入组取出第一个元素。  

    j=i-1;//i-1即为有序组最后一个元素(与待插入元素相邻)的下标  

    while(j>=0&&tmp<a[j])//注意判断条件为两个,j>=0对其进行边界限制。第二个为插入判断条件  

    {

    a[j+1]=a[j];//若不是合适位置,有序组元素向后移动  

    j--;

    }

    a[j+1]=tmp;//找到合适位置,将元素插入。  

    }

    }

(四)希尔排序

算法思想 : 将无序数组分割为若干个子 序列,子序列不是逐段分割的,而是相隔特定的增量的子序列,对各个子序列进行插入排序;然后再选择一个更小的增量,再将数组分割为多个子序列进行排序......最后选择增量为1,即使用直接插入排序,使最终数组成为有序。

时间复杂度 :O(n*log(n))~O(n^2) ,最坏情况为O(n^2)

    // 希尔排序

    publicstaticvoidshell_sort(intarr[],intsize) {

        if(arr==null)

            return;

        inth=1;/* 关于步长,取值没有统一标准,必须小于size,最后一次步长要为1 */

        /* 计算首次步长 */

        while(h<size/3)

            h=3*h+1;

        inti,j,temp;

        while(h>=1) {

            for(i=h;i<size;++i) {

                /* 将a[i]插入到a[i-h]、a[i-2h]、a[i-3h]...中 */

                for(j=i;j>=h&&(arr[j]<arr[j-h]);j-=h) {

                    temp=arr[j];

                    arr[j]=arr[j-h];

                    arr[j-h]=temp;

                }

            }

            /* 计算下一轮步长 */

            h=h/3;

        }

    }

(五)快速排序

快速排序 :选取一个数,位置i,进行排序比较,从j开始,选择比选取的数小的,填补到原来的位置,则此时j位置,空出来,再从i++开始,选择比i大的数,填补到j的位置,重复排序,则是大的放右边,小的左边,然后继续递归调动。

时间复杂度 :O(N*logN)

//快速排序

voidquick_sort(ints[],intl,intr)

{

if(l<r)

   {


inti=l,j=r,x=s[l];

while(i<j)

       {

while(i<j&&s[j]>=x)// 从右向左找第一个小于x的数

                j--;

if(i<j)

                s[i++]=s[j];


while(i<j&&s[i]<x)// 从左向右找第一个大于等于x的数

                i++;

if(i<j)

                s[j--]=s[i];

       }

s[i]=x;

quick_sort(s,l,i-1);// 递归调用

quick_sort(s,i+1,r);

   }

}

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

推荐阅读更多精彩内容

  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 3,320评论 0 2
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 5,164评论 0 52
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 1,235评论 0 2
  • 导语 许多人走出家乡或许都是因为大学,离开父母,离开自己熟悉的地方。 我来自广西,一个普通的地方,一个普通的少年,...
    一个十八Wayne_w阅读 1,125评论 2 9
  • 雪似细雨舞斜风,梅似佳人傍苍松。 独坐窗前一壶酒,喜看红玉傲寒冬。 一丛兰雅和: 斗雪迎春早,何惧红颜老? 劲骨临...
    简村小吹阅读 1,021评论 17 17