排序一:冒泡、插入、选择

文章结构

  1. 概述
  2. 冒泡排序
  3. 插入排序
  4. 选择排序

1. 概述

常见的排序算法有:冒泡排序、插入排序、选择排序、归并排序、快速排序、计数排序、基数排序、桶排序

衡量排序算法的标准

  1. 时间复杂度
  2. 空间复杂度
  3. 是否稳定

按照时间复杂度,可以将上面的算法划分为

时间复杂度 算法名称
O(n^2) 冒泡、插入、选择
O(nlogn) 归并、快排
O(n) 计数、基数、桶

这篇文章我们先来看一下几个时间复杂度为O(n^2)的排序算法:冒泡排序、插入排序、选择排序

2. 冒泡排序

2.1 什么是冒泡排序

冒泡排序的思想就是:比较相邻两个元素的大小,如果满足大小关系,则用第二个元素和下一个元素继续比较;如果不满足大小关系,则交换两个元素的位置,然后用交换后的第二个元素和下一个元素继续比较,依次比较到最后一个元素。通过这样一轮冒泡排序之后,最大的元素就到了最后的那个位置上。再次重复这个过程将次大的元素排好,重复n次则所有的元素都排好了

优化技巧:如果某一轮冒泡排序中,没有元素需要交换,则整个待排序的数据已经有序了,则不需要再进行排序了。

2.2 动画演示

冒泡排序.gif

2.3 过程分析

我们来分析一轮冒泡排序的过程,使用红色标记相邻两个待比较的元素

冒泡排序过程分析.png

第一轮比较结束,我们将待排序列表中的最大元素45排到了最后面。然后我们继续按照这个思路,对前面的n-1个元素继续排序。

2.3 代码实现

public static void sort(int[] array) {
    if (array == null || array.length == 1) {
        return;
    }
    int size = array.length;
    for (int i = 0; i < size; i++) {
        boolean isExchanged = false;
        for (int j = 0; j < size - i - 1; j++) {//这里的i表示经过了几次冒泡排序了,经过一次有在最后面多了一位已经排好序的元素,所以下一次要比较的范围也较少1
            if (array[j] > array[j + 1]) {//只有在后面的元素大于前面的元素的时候才交互,可以保证排序的稳定性
                int temp = array[j + 1];
                array[j + 1] = array[j];
                array[j] = temp;
                isExchanged = true;
            }
        }
        if (!isExchanged) {//当某一趟冒泡排序中没有交换发送的时候,说明序列已经有序了
            break;
        }
    }
}

2.4 性能分析

时间复杂度分析

  1. 最好时间复杂是O(n):当待排序的列表本身的顺序和要求的顺序一致的时候,我们只需要进行一轮冒泡排序,发现没有数据需要交换,我们就知道列表已经满足要求的顺序了,不需要再继续排序,此时时间复杂度为O(n)
  2. 最差时间复杂度O(n^2):当待排序的列表本身的顺序和要求的顺序完全相反的时候,我们需要进行n-1轮冒泡排序
  3. 平均时间复杂度O(n^2)

空间复杂度

冒泡排序整个排序过程中,只需要借助一个额外的辅助空间,所以空间复杂度为O(1),我们把这种空间复杂度为O(1)的排序叫做原地排序

是否稳定

什么是稳定排序:稳定排序排序就是排序前值相同的元素之间的相对位置和排序之后的相对位置一样。比如有两个元素的值都是14,如果排序之后,原来在前面的14,现在排到了后面去了,那就不是稳定排序;如果依然排在前面,则是稳定排序

冒泡排序是稳定排序:冒泡排序过程中只是对相邻元素进行交换,我们可以控制当相邻两个元素的值一致时,不进行交互,这样就可以保证稳定性

3. 插入排序

3.1 什么是插入排序

插入排序的思想是将序列分为两个:一个有序序列,一个无需序列;然后遍历无序序列,依次取出元素插入到有序序列中的合适位置,使得有序序列依然有序,无序序列遍历完了,整个序列就都有序了

3.2 动画演示

插入排序.gif

3.3 过程分析

我们来分析一下插入排序的过程,有序序列我们用红色标记,刚刚插入的值我们使用蓝色标记


插入排序过程分析.png

3.4 代码实现

public static void sort(int[] a) {
    if (a == null || a.length == 0) {
        return;
    }
    int size = a.length;
    for (int i = 1; i < size; i++) {
        int temp = a[i];
        int index = i - 1;
        while (index >= 0 && temp < a[index]) {//从后向前比较,寻找插入元素的位置,只有小于的时候,才移动原来的序列,保证排序的稳定性
            a[index + 1] = a[index];
            index--;
        }
        a[index + 1] = temp;
    }
}

3.5 性能分析

时间复杂度

  1. 最好时间复杂是O(n):当待排序的列表本身的顺序和要求的顺序一致的时候,我们只需进行一轮遍历,不需要查找插入位置和执行插入操作
  2. 最差时间复杂度O(n^2):当待排序的列表本身的顺序和要求的顺序完全相反的时候,每一个元素都需要和有序序列中所有的元素进行比较已查找插入的位置,并执行插入操作
  3. 平均时间复杂度O(n^2)

空间复杂度

也只需要借助一个辅助空间,所以空间复杂度为O(1)

是否稳定

插入排序过程中,在查找插入位置的时候,控制好不要将后面与前面值相等的元素插入到相同值的前面就可以保证稳定性。所以插入排序是稳定的。

4. 选择排序

4.1 什么是选择排序

选择排序也是将序列分为两个序列:一个有序(初始化为空),一个无序(待排序的序列)。然后从无序序列中选择最小的元素放到有序序列的最后(和无序序列中的第一个元素交换位置),直到无序序列中没有元素

4.2 动画演示

选择排序.gif

4.3 过程分析

我们依然29,10,14,37,14,45,13,22,11这样一个待排序序列来分析一下排序过程。有序序列我们用红色标记,无序序列的最小值,我们用蓝色标记,无序序列中要交换的位置我们用橙色标记


选择排序过程分析.png

4.4 代码实现

public static void sort(int[] array) {
    if (array == null || array.length == 0) {
        return;
    }
    int size = array.length;
    for (int i = 0; i < size - 1; i++) {
        //从无序序列中找到最小的元素
        int index = i;
        for (int j = i + 1; j < size; j++) {
            if (array[index] > array[j]) {
                index = j;
            }
        }
        //无序序列中的最小值和无序序列中的第一个元素交换位置(因为这里的交互元素的位置,所以选择排序不是稳定排序)
        int temp = array[i];
        array[i] = array[index];
        array[index] = temp;
    }
}

4.5 性能分析

时间复杂度

无论是否有序,从无序序列中选择最小元素都需要经过n次比较,所以选择排序的最好、最差、平均时间复杂度都是O(n^2)

空间复杂度

选择排序也只需要借助一个额外的辅助空间,所以选择排序的时间复杂度也是O(1)

是否稳定

选择排序不是稳定排序,因为选择排序的过程中存在将无序序列最小的元素和无序序列第一个元素交换的操作,可能把相同值的前一个元素交换到后面去了。

源码

源码和测试用例请查看github之排序

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