算法学习02_初等排序

“有匪君子,如切如磋,如琢如磨 ”

概述

冒泡排序、选择排序、插入排序三种排序算法的时间复杂度为O(n^2),经常被用来作为排序算法的入门。虽然这三种排序算法相对于其他高级排序算法时间效率较低,但是仍然有学习的价值。

  • 实际系统应用的算法都是经过综合优化后,初等排序算法可以用作高级排序算法的优化的子过程,例如插入排序常常被用作小规模算法的实现。
  • 初等算法的实现过程的方法思想可以被借鉴。

初等排序算法共同思路是把待排序的序列分为已排序部分(如下绿色部分)和未排序(如下红色部分)部分,然后对未排序部分处理,将其逐个合并到已经排序部分,差异之处在于合并的方法。


排序.png

冒泡

思想

核心思想是从序列的一端开始寻找第i大,将其放置于最终排序的位置,寻找第i个元素的过程是不停的交换,如下图所示。

冒泡排序动态演示

实现

冒泡排序的算法具体实现,根据每次冒泡找到的是最小还是最大有两种实现方式。

  • 寻找最小者
    static void bubble_min(T arr[], int size) {
        for (int i = 0; i < size; i++) {
            for (int j = size - 1; j > 0; j--) {
                if (arr[j] < arr[j - 1]) {
                    swap(arr[j], arr[j - 1]);
                }
            }
        }
    }
  • 寻找最大者
static void bubble_max(T arr[], int size) {
        for (int i = size-1; i > 0; i--) {
            for (int j = 0; j < i; j++) {
                if (arr[j] > arr[j + 1]) {
                    swap(arr[j], arr[j + 1]);
                }
            }
        }
    }
  • 边界值
    边界值选取直接决定实现的算法是否正确,边界值的选取依赖于对于算法思想的理解和对于每个变量的定义。
    寻找最小者实现中:外层循环变量i的含义是每次冒泡确定下来的最小元素在数组中的下标,想要对整个数组实现排序,小元素的下标小,该变量应该覆盖数组全部元素,所以i的范围应该是[0, size-1];变量j的含义是内层循环过程用来交换两个元素的下标较大者,按照这种定义j的范围应该是[size-1, i),下面的比较、交换的元素就是j,j-1,这里需要注意的是变量含义的定义和后续的实现应该匹配,只要内在的逻辑自洽即可。
    寻找最大者实现中:外层循环变量i的含义是每次冒泡确定下来的最大元素在数组中的下标。想要对整个数组实现排序,小元素的下标小,该变量应该覆盖数组全部元素,所以i的范围应该是[size-1, 0];变量j的含义是内层循环过程用来交换两个元素的下标较小者,按照这种定义j的范围应该是[0, i),下面的比较、交换的元素就是j,j+1,这里需要注意的是变量含义的定义和后续的实现应该匹配,只要内在的逻辑自洽即可。
  • 实测用时
    冒泡排序的时间复杂度是O(n^2),通过实测记录其用时曲线如下,类似一条抛物线。
    冒泡排序用时

优化

如果内层循环在一轮下来已经没有发生交换,说明“未排序”部分已经有序,则排序循环可以终止,实现如下。

    static void bubble_op(T arr[], int size) {
        for (int i= 0; i < size - 1; i++) {
            bool swap_flag = false;
            for (int j = size - 1; j > 0; j--) {
                if (arr[j] < arr[j - 1]) {
                    swap(arr[j], arr[j - 1]);
                    swap_flag = true;
                }
            }
            if (false == swap_flag) {
                break;
            }
        }
    }

选择

思想

冒泡排序的是通过把较小者不停向前交换达到将无序部分合并到有序部分,其实我们需要达到的目的是找到最小者,所以是否可以避免无用的交换,只需在找到最小者后将其交换到合适的位置即可。


选择排序动态演示

实现

使用一个变量记录内层循环每次找到的最小索引,内层循环结束后将交换到合适位置。

    static void select(T arr[], int size) {
        for (int i = 0; i < size; i++) {
            int min = i;
            for (int j = i + 1; j < size; j++) {
                if (arr[j] < arr[min]) {
                    min = j;
                }
            }
            swap(arr[i], arr[min]);
        }
    }
  • 边界
    外层循环需要遍历到数组所有元素所以i范围是[0,size-1],内层循环在[i+1, size-1]范围内搜索最小者。

插入

思想

image.png

插入排序类似在玩扑克牌的时候手中牌已经有序,从牌堆中抓取一张牌,然后将其插入到手中合适的位置。插入排序的优点是可以提前完成内层循环,即对于近乎有序序列可以效率很高。

实现

    static void insert(T arr[], int size){
        // i 待插入元素
        for (int i = 1; i < size; i++) {
            // j 待考察的插入位置
            for (int j = i - 1; j >= 0; j--) {
                if (arr[j + 1] < arr[j]) {
                    swap(arr[j + 1], arr[j]);
                }
                else {
                    break;
                }
            }
        }
    }

边界

外层循环变量i指示待插入的元素,认为第一个元素只有其自身,已经有序,所以i的范围是[1, size-1]。内层循环变量j用来表示下一个待插入的位置,其范围是[i-1, 0],内层循环比较的时候应该将当前插入位置j+1和下一个插入位置j元素比较。

优化

之前的实现方式中的内层循环中,如果后一个元素小于前一个元素则需要不停的交换两者,以达到找寻合适位置的目的。由于前边有序部分的已经完成排序,现在只是需要将新考察的元素插入。可以将考察元素依次和有序元素比较,如果待考察元素小于有序元素则将有序元素后移,否则待考察元素已经找到合适位置,如下动图所演示。


插入排序动态演示

具体代码如下

    static void insert_op(T arr[], int size) {
        for (int i = 1; i < size; i++) {
            int j = i - 1;
            T   v = arr[i];
            for (; (j >= 0) && (arr[j] > v); j--) {
                arr[j + 1] = arr[j];
            }
            arr[j + 1] = v;
        }
    }

希尔

思想

插入排序的优势在处理序列元素个数较少并且近乎有序的序列,但是极端情况下对于逆序序列内层循环每次都需要全部执行完,这样插入插入排序的时间复杂度就退化成O(n^2).Donald Shel 1959年提出希尔(shell)排序算法,其核心思想是对待排序序列使用固定间隔分成若干组,对每一个分组内的元素使用插入排序,然后变更固定间隔,最后一次以1为间隔执行插入排序。希尔排序是先在总体让序列有序,然后在一步步细化。希尔排序对使用的间隔序列对输入序列规模有依赖,通常选用\{3*n+1|n<size,n=0,1,2..\}序列。

实现

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

推荐阅读更多精彩内容

  • 搞懂基本排序算法 上篇文章写了关于 Java 内部类的基本知识,感兴趣的朋友可以去看一下:搞懂 JAVA 内部类;...
    醒着的码者阅读 1,177评论 3 4
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,856评论 25 707
  • 1、对个人能力的感知力---“我能行” 2、对自己在重要关系中的价值的感知力---“我的贡献有价值,大家确实需要我...
    艾葭_阅读 352评论 0 0
  • 凉凉夜色 天空如墨 星子在其间闪烁 没有月亮 缺少思念 一个人走在街道旁 没有孤寂 亦没有热闹 心似湖水 表面的平...
    爱丽丝寻梦之旅阅读 223评论 0 1