排序算法

张俊玲

19021210858

转载自:https://blog.csdn.net/qq_36511401/article/details/102638452

【嵌牛导读】排序的时间和空间复杂度是检验一个算法是否优秀的标准。

【嵌牛鼻子】直接插入排序   半插入排序   希尔排序 

【嵌牛提问】排序是否稳定?

【嵌牛正文】

一、直接插入排序。

1、介绍。

        直接插入排序是一种简单的插入排序法,其基本思想是:把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列。例如,已知待排序的一组记录是:60,71,49,11,24,3,66。假设在排序过程中,前3个记录已按关键码值递增的次序重新排列,构成一个有序序列:49,60,71。将待排序记录中的第4个记录(即11)插入上述有序序列,以得到一个新的含4个记录的有序序列。首先,应找到11的插入位置,再进行插入。可以将11放入数组的第一个单元r[0]中,这个单元称为监视哨,然后从71起从右到左查找,11小于71,将71右移一个位置,11小于60,又将60右移一个位置,11小于49,又再将49右移一个位置,这时再将11与r[0]的值比较,11≥r[0],它的插入位置就是r[1]。假设11大于第一个值r[1]。它的插入位置应该在r[1]和r[2]之间,由于60已经右移了,留出来的位置正好留给11.后面的记录依照同样的方法逐个插入到该有序序列中。若记录数n,续进行n-1趟排序,才能完成。 

        直接插入排序是稳定排序,不需要额外内存,空间复杂度O(1)。时间复杂度,最佳情况:O(n)  最差情况:O(n^2)  平均情况:O(n^2)。

2、步骤。

(1)设置监视哨r[0],将待插入记录的值赋值给r[0];

(2)设置开始查找的位置j;

(3)在数组中进行搜索,搜索中将第j个记录后移,直至r[0].key≥r[j].key为止;

(4)将r[0]插入r[j+1]的位置上。

3、代码。

public static void main(String[] args) {

        System.out.println("------开始------");

        //生成生成两份一模一样的随机数组,其中一组用系统自带的方法进行排序,到时候进行验证。

        final int number = 100000;

        int[] sortArray = new int[number];

        int[] sortArrayCopy = new int[number];

        for (int i = 0; i < sortArray.length; i++) {

            sortArray[i] = (int) (Math.random() * number);

        }

        System.arraycopy(sortArray, 0, sortArrayCopy, 0, number);//数组复制

        Arrays.sort(sortArrayCopy);

        //开始排序

        long startTime = System.currentTimeMillis();

        directInsertSort(sortArray);//直接插入排序

        System.out.println("花费时间:" + (System.currentTimeMillis() - startTime));

        //跟系统排序之后数组进行比较,查看是否排序成功。

        if (Arrays.equals(sortArray, sortArrayCopy)) {

            System.out.println("排序成功");

        } else {

            System.out.println("排序失败");

        }

        System.out.println("------结束------");

    }

//直接插入排序 最佳情况:T(n) = O(n) 最坏情况:T(n) = O(n2) 平均情况:T(n) = O(n2)

private static void directInsertSort(int[] array) {

    for (int i = 1; i < array.length; i++) {//n-1轮  第一个无需排序

        int curIndex = i;

        while (curIndex > 0) {

            if (array[curIndex] > array[curIndex - 1]) {

                break;

            }

            int flag = array[curIndex];

            array[curIndex] = array[curIndex - 1];

            array[curIndex - 1] = flag;

            curIndex--;

        }

    }

}

4、结果。

二、折半插入排序。

1、介绍。

        将直接插入排序中寻找A[i]的插入位置的方法改为采用折半比较,即可得到折半插入排序算法。在处理A[i]时,A[0]……A[i-1]已经按关键码值排好序。所谓折半比较,就是在插入A[i]时,取A[i-1/2]的关键码值与A[i]的关键码值进行比较,如果A[i]的关键码值小于A[i-1/2]的关键码值,则说明A[i]只能插入A[0]到A[i-1/2]之间,故可以在A[0]到A[i-1/2-1]之间继续使用折半比较;否则只能插入A[i-1/2]到A[i-1]之间,故可以在A[i-1/2+1]到A[i-1]之间继续使用折半比较。如此担负,直到最后能够确定插入的位置为止。一般在A[k]和A[r]之间采用折半,其中间结点为A[k+r/2],经过一次比较即可排除一半记录,把可能插入的区间减小了一半,故称为折半。执行折半插入排序的前提是文件记录必须按顺序存储。 

        折半插入排序是稳定排序,不需要额外内存,空间复杂度O(1)。时间复杂度,最佳情况:O(n^2)  最差情况:O(n^2)  平均情况:O(n^2)。

2、步骤。

        跟直接插入排序的步骤相似,只不过查找插入点的方法不一样。直接插入排序是从有序区的最后一个数依次向前找,而折半插入排序是通过折半的方式进行查找。

(1)计算 0 ~ i-1 的中间点,用 i 索引处的元素与中间值进行比较,如果 i 索引处的元素大,说明要插入的这个元素应该在中间值和刚加入i索引之间,反之,就是在刚开始的位置 到中间值的位置,这样很简单的完成了折半;

(2)在相应的半个范围里面找插入的位置时,不断的用(1)步骤缩小范围,不停的折半,范围依次缩小为 1/2 1/4 1/8 .......快速的确定出第 i 个元素要插在什么地方;

(3)确定位置之后,将整个序列后移,并将元素插入到相应位置。

3、代码。

public static void main(String[] args) {

        System.out.println("------开始------");

        //生成生成两份一模一样的随机数组,其中一组用系统自带的方法进行排序,到时候进行验证。

        final int number = 100000;

        int[] sortArray = new int[number];

        int[] sortArrayCopy = new int[number];

        for (int i = 0; i < sortArray.length; i++) {

            sortArray[i] = (int) (Math.random() * number);

        }

        System.arraycopy(sortArray, 0, sortArrayCopy, 0, number);//数组复制

        Arrays.sort(sortArrayCopy);

        //开始排序

        long startTime = System.currentTimeMillis();

        halveInsertSort(sortArray);//折半插入排序

        System.out.println("花费时间:" + (System.currentTimeMillis() - startTime));

        //跟系统排序之后数组进行比较,查看是否排序成功。

        if (Arrays.equals(sortArray, sortArrayCopy)) {

            System.out.println("排序成功");

        } else {

            System.out.println("排序失败");

        }

        System.out.println("------结束------");

    }

//折半插入排序 最佳情况:T(n) = O(n) 最坏情况:T(n) = O(n2) 平均情况:T(n) = O(n2)

private static void halveInsertSort(int[] array) {

    int flag;

    int low, middle, high;

    for (int i = 1; i < array.length; i++) {//n-1轮

        flag = array[i];

        low = 0;

        high = i - 1;

        while (low <= high) {//最后的情况就是low==high==middle的判断

            middle = (low + high) / 2;

            if (array[i] > array[middle]) {

                low = middle + 1;

            } else {

                high = middle - 1;

            }

        }

        for (int j = i; j > high + 1; j--) {

            array[j] = array[j - 1];

        }

        array[high + 1] = flag;

    }

}

4、结果。

三、希尔排序。

1、介绍。

       希尔排序(Shell's Sort)是插入排序的一种又称“缩小增量排序”(Diminishing Increment Sort),是直接插入排序算法的一种更高效的改进版本。希尔排序是非稳定排序算法。该方法因D.L.Shell于1959年提出而得名。希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。

        希尔排序是不稳定排序,不需要额外内存,空间复杂度O(1)。时间复杂度,最佳情况:O(nlog^2n)  最差情况:O(nlog^2n)  平均情况:O(nlogn)。

2、步骤。

        我们来看下希尔排序的基本步骤,在此我们选择增量gap=length/2,缩小增量继续以gap = gap/2的方式,这种增量选择我们可以用一个序列来表示,{n/2,(n/2)/2...1},称为增量序列。希尔排序的增量序列的选择与证明是个数学难题,我们选择的这个增量序列是比较常用的,也是希尔建议的增量,称为希尔增量,但其实这个增量序列不是最优的。此处我们做示例使用希尔增量。先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,具体算法描述:选择一个增量序列t1,t2,…,tk,其中ti>tj,tk=1;按增量序列个数k,对序列进行k 趟排序;每趟排序,根据对应的增量ti,将待排序列分割成若干长度为m 的子序列,分别对各子表进行直接插入排序。仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。

3、代码。

public static void main(String[] args) {

        System.out.println("------开始------");

        //生成生成两份一模一样的随机数组,其中一组用系统自带的方法进行排序,到时候进行验证。

        final int number = 100000;

        int[] sortArray = new int[number];

        int[] sortArrayCopy = new int[number];

        for (int i = 0; i < sortArray.length; i++) {

            sortArray[i] = (int) (Math.random() * number);

        }

        System.arraycopy(sortArray, 0, sortArrayCopy, 0, number);//数组复制

        Arrays.sort(sortArrayCopy);

        //开始排序

        long startTime = System.currentTimeMillis();

        shellInsertSort(sortArray);//希尔插入排序

        System.out.println("花费时间:" + (System.currentTimeMillis() - startTime));

        //跟系统排序之后数组进行比较,查看是否排序成功。

        if (Arrays.equals(sortArray, sortArrayCopy)) {

            System.out.println("排序成功");

        } else {

            System.out.println("排序失败");

        }

        System.out.println("------结束------");

    }

//希尔插入排序 最佳情况:T(n) = O(nlog2 n) 最坏情况:T(n) = O(nlog2 n) 平均情况:T(n) =O(nlog2n)

private static void shellInsertSort(int[] array) {

    int groups = array.length / 2;//增量,一共的组数

    while (groups > 0) {

        //将groups看作1,就会跟直接插入排序的算法一模一样

        for (int i = groups; i < array.length; i++) {//n-1轮  第一个无需排序

            int curIndex = i;

            while (curIndex > groups - 1) {

                if (array[curIndex] > array[curIndex - groups]) {

                    break;

                }

                int flag = array[curIndex];

                array[curIndex] = array[curIndex - groups];

                array[curIndex - groups] = flag;

                curIndex -= groups;

            }

        }

        groups /= 2;

    }

}

4、结果。

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