Swift算法俱乐部中文版 -- 插入排序

目标:从小到大(或从大到小)对数组进行排序。

给你一组数组,将他们排序。插入排序算法的步骤如下:

  • 把未排序的数字放在一堆。

  • 从这堆数字里面挑一个。选哪个并不重要,但从顶部拿最容易。

  • 将选出来的数字插入新数组中。

  • 从未排序的数字堆中挑一个数字,并将其插入新数组。 在您选择的第一个数字之前或之后,将这两个数字排序。

  • 再选择一个数字,并按顺序插入到数组中适当的位置。

  • 重复这样做,直到堆里没有数字。 你最终得到一个空堆和一个排序的数组。

这就是为什么这被称为“插入”排序,因为你从堆中取一个数字,并将其插入在数组里正确的排序位置。

我们摸扑克牌就是插入排序!

举个栗子


未排序的数字堆是 [ 8, 3, 5, 4, 6 ]

选择第一个数字 8 ,并将其插入新数组。 新数组是空的,所以很容易。 排序的数组现在是 [8] ,未排序的堆是 [ 3, 5, 4, 6 ]

从堆中选择下一个数字 3 ,并将其插入到排序的数组中。 它应该在8之前,所以排序的数组现在 [ 3, 8 ] ,堆是 [ 5, 4, 6 ]

从堆中选择下一个数字5,并将其插入到排序的数组中。 它在38之间。排序的数组是 [ 3, 5, 8 ] ,堆是 [ 4, 6 ]

重复此过程,直到堆是空的。

在一个数组中排序


上面的解释会你看起来需要两个数组:一个用于未排序的数组,一个排序好的数组。

但是你可以在一个数组里执行插入排序,而无需创建新的数组。 你只需知道数组的哪个部分已经排序,哪个部分是未排序的堆。

最初,数组是 [ 8, 3, 5, 4, 6 ] 。用 | 区别排序部分和未排序部分:

[| 8, 3, 5, 4, 6 ]

这表示排序部分是空的,未排序的数字堆从8开始。

处理完第一个数字后,是这样的:

[ 8 | 3, 5, 4, 6 ]

排序部分是 [ 8 ] ,堆是 [ 3, 5, 4, 6 ]| 已经向右移位了一位。

这是数组的排序过程:

[| 8, 3, 5, 4, 6 ]
[ 8 | 3, 5, 4, 6 ]
[ 3, 8 | 5, 4, 6 ]
[ 3, 5, 8 | 4, 6 ]
[ 3, 4, 5, 8 | 6 ]
[ 3, 4, 5, 6, 8 |]

在每个步骤中,| 从数组的开头,一下一下的向后移动,直到排序完成时到达数组的结尾。 数字堆一个个缩小,并且排序部分一个个增加,直到数字堆是空的,并且所有数字都排序完成。

怎样插入?


每个步骤,都要从未排序的堆中选择最顶层的数字,并将其插入排序数组。 您必须把数字放在合适的位置,保证数组是排序好的。 怎样做呢?

我们假设已经做了几步,数组看起来像这样:

[ 3, 5, 8 | 4, 6 ]

下一个要排序的数字是 4 ,排序好的部分是 [ 3, 5, 8 ]

用这种方法:看看上一个元素 8

[ 3, 5, 8, 4 | 6 ]
        ^

8 是否大于 4 ? 是的,所以 4 应该在 8 之前。我们交换这两个数字得到:

[ 3, 5, 4, 8 | 6 ]
        <-->
        交换

还没完。 新的前一个元素 5 也大于 4 。我们还交换这两个数字:

[ 3, 4, 5, 8 | 6 ]
     <-->
     交换

再看看前面的元素。 3 大于 4 吗? 不是。 这意味着我们完成了 4 在数组中的排序。

这是插入排序算法,您将在下一节中看到内部循环的描述部分:通过交换数字将从堆的顶部的数字插入到排序的部分。

代码


这是用 Swift 实现的插入排序:

func insertionSort(_ array: [Int]) -> [Int] {
    var a = array                           // 1
    for x in 1..<a.count {                  // 2
        var y = x
        while y > 0 && a[y] < a[y - 1] {    // 3
            swap(&a[y - 1], &a[y])
            y -= 1
        }
    }
    return a
}

把这段代码放到 playground 里测试:

let list = [ 10, -1, 3, 9, 2, 27, 8, 5, 1, 3, 0, 26 ]
insertionSort(list)

注释对应的代码分析:

  1. 创建数组的副本。这是必要的,因为我们不能直接修改参数 array 的内容。 像Swift 中的 sort()一样,insertionSort() 函数将返回原始数组的副本,并将它排序。

  2. 这个函数里有两个循环:外部循环遍历数组中的每个元素,从数字堆的最顶端挑选数字;变量 x 是排序区分排序不分与数字堆的索引(相当于 | 的作用)。请记住,任何时候,array[0]array[x] 都是排序完成的。从 array[x] 到结束,是未排序的数字堆。

  3. 循环查看 array[x] ,这是在数字堆中顶部的数组,它可能小于排序数组的任何元素。循环排序好的数组,每次往前找一个,如果前面的更大,就交换它们。当循环完成时,排序好的数组就增长了一个元素。

注意:外循环从(注释2处)索引从1开始,将第一个元素从数字堆中移动到排序部分并不会改变任何内容,所以我们可以跳过它。

减少交换


上面的插入排序虽然能使用,但是如果能删除 swap() 方法会运行的更快。

您已经看到,我们交换数字,将下一个元素根据排序移动到合适的位置:

[ 3, 5, 8, 4 | 6 ]
        <-->
        swap

[ 3, 5, 4, 8 | 6 ]
     <-->
     swap

我们可以不交换,将数字向右移动,然后把新数字放到合适的位置就可以了。

[ 3, 5, 8, 4 | 6 ]  复制 4
           *

[ 3, 5, 8, 8 | 6 ]  将 8 向右移动
        --->

[ 3, 5, 5, 8 | 6 ]  将 5 向右移动
     --->

[ 3, 4, 5, 8 | 6 ]  把 4 粘贴到合适的位置
     *

代码:

func insertionSort(_ array: [Int]) -> [Int] {
    var a = array
    for x in 1..<a.count {
        var y = x
        let temp = a[y]
        while y > 0 && temp < a[y - 1] {
            a[y] = a[y - 1]                 // 1
            y -= 1
        }
        a[y] = temp                         // 2
    }
    return a
}

// 1 处是将数字向右移动一位。内循环结束时,y 是排序数组中和新数字比较的索引。// 2 处是将新数字复制到对应的位置。

给其他类型排序


我们可以给其他类型排序,添加泛型并提供比较函数即可。只需要修改两处代码。

方法定义变为:

func insertionSort<T>(_ array: [T], _ isOrderedBefore: (T, T) -> Bool) -> [T] {

数组具有泛型 [T],现在 isOrderedBefore() 方法将接受任何类型,无论是数组,字符串,或是其他类型。

新的参数 isOrderedBefore: (T, T) -> Bool 是一个方法 ,接受两个 T 类型的对象,如果两个参数比较厚满足这个表达式,返回 true,不满足则返回 false。就像是 Swift 内置的 sort() 方法。

另一个改动的地方是在内循环:

    while y > 0 && isOrderedBefore(temp, a[y - 1]) {

temp < a[y - 1] 替换为 isOrderedBefore() 方法。它做了同样的事情,并且可以比较任何类型。

在 Playground 中测试:

let numbers = [ 10, -1, 3, 9, 2, 27, 8, 5, 1, 3, 0, 26 ]
insertionSort(numbers, <)
insertionSort(numbers, >)

<> 决定排序顺序,分别是从小到大和从大到小。

当然,你也可以排序其他类型,如字符串:

let strings = [ "b", "a", "d", "c", "e" ]
insertionSort(strings, <)

甚至是更复杂的对象:

let objects = [ obj1, obj2, obj3, ... ]
insertionSort(objects) { $0.priority < $1.priority }

闭包是告诉 insertionSort() 根据对象的属性 priority 进行排序。

插入排序是一种稳定的排序。对于都具有排序键的元素,是相对稳定的。这对于简单的数字或字符串不重要 ,但对于复杂对象就非常重要了。在上面的例子中,两个对象的 priority 一样的话,就不会交换,不管其他属性的值多大多小。

性能


如果数组已经排序,那插入排序真的很快。这听起来很明显,但不是所有的排序算法都能做到。在实践中,如果是大量数据并不完全排序,插入排序会表现的很好。

最坏情况下插入排序的平均情况性能是 O(n ^ 2) 。 这是因为在这个函数中有两个嵌套循环。 其他排序算法(如快速排序和合并排序)的性能是 O(n log n),在数据量较大时更快。

插入排序实际上对排序小数组非常快。 当元素个数为10或更小需要排序的时候,一些标准库会把它们的排序方法从快速排序切换到插入排序。

我做了一个简单的测试比较我们的 insertionSort() 与 Swift 的内置 sort() 。 在大约100个元素左右的数组上,速度的差异很小。 然而,随着个数增加,O(n ^ 2) 很快开始执行比 O(n log n) 差很多,并且插入排序不能跟上。

作者:Matthijs Hollemans -- Swift 算法俱乐部

英文链接:
https://github.com/raywenderlich/swift-algorithm-club/tree/master/Insertion%20Sort

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

推荐阅读更多精彩内容