使用OC写算法之插入排序

序言

看了一下最近写的OC写算法系列,好像还有不少排序算法都没有更新,可能有的朋友也有这方面的需求,所以最近我会把插入排序、归并排序、快速排序等更新出来,好吧,下面废话不多说,咱们进入正题。

什么是插入排序

先来简单的说句大白话,插入排序就是通过找到元素最合适插入位置的方式来进行排序的一中排序,下面再聊一下官方介绍:

插入排序是指每次将一个待排序的记录,按其关键字大小插入到前面已经排好序的子序列中的适当位置,直到全部记录插入完成为止。

下面是我从网上扣的一张图,我觉得这张图描述的很清晰,下面请看图:

插入排序示意图.png

我们可以看到假如我们要找到4这个元素合适的插入位置,我们需要拿它与前面已经排好序的元素一次进行比较,当我们拿4和7进行比较后发现,4 比 7 小,然后我们就让他们交换位置,这是4和7已经是有序的了,这时候我们来寻找一下-2这个元素与前面已经排好序的元素去比较,发现-2比7小,交换位置,再与4比较发现比4小,此时它已经是第0个元素了,所以第0个位置就是它要插入的位置了,依次类推,你会发现经历5次寻找插入位置的操作过后,数组已经变成完全有序了,下面我们来看代码实现:

#pragma  mark -  未优化的插入排序
#pragma  mark -
/**
 未进行优化的插入排序(多次交换性能差)

 @param array 未排序数组
 */
- (void)insertionSort1:(NSMutableArray *)array {
    //i从1开始,因为第0个元素默认就是有序的
    for (int i = 1 ; i < array.count; i++) {
        //j从i这个位置开始 依次-- 和i前面的元素进行比较 寻找合适的插入位置 交换位置
        for (int j = i; j > 0 && array[j] < array[j - 1]; j--) {
            //每次比较当前元素和前一个元素 当前元素大于前一个时交换位置
            [array exchangeObjectAtIndex:j withObjectAtIndex:j -1];
        }
    }
}

我们这时可以测试一下50000的数据,执行效果和代码执行时长:

    //生成5万个元素的随机数组
    NSMutableArray *array1 = [self.testHelper generateRandomArray:50000 rangeLeft:1 rangeRight:100];

    NSLog(@"=============未优化的插入排序耗时==============");
    [self.testHelper testSortWithExcuteBlock:^{
        [self insertionSort1:array1];
    }];
    NSLog(@"%@",array1);

下面是测试结果:

未优化插入排序.png
插入排序的优化

我们发现耗时是相对比较长的,那到底是什么原因呢?从代码上看,我们就知道每一次插入都有多次交换操作,而一次交换操作相当于三次赋值操作,这是比较耗费性能的,那么我们有没有优化的方法呢?答案当然是有的,下面我们看下面的图,


插入排序优化图1.png

假如我们需要将下面的一个数组进行排序,我们从6这个元素开始考察,按照原来插入排序的做法是,发现与8比较要小于8,所以两个交换一下位置,但是我们这里不这么做,而是将这个元素复制一份假如就叫 temp,看下图:

插入排序优化图2.png

那么我们就可以拿temp这个元素与前面排好序的元素进行比较,我们发现temp比8小,此时我们将8往后挪动一个位置,再来考察8前面的元素与6进行比较,发现已经是第0个元素了,所以我们直接就可以将temp这个元素放入到8原来的位置上了,


插入排序优化图3.png

经历了上面的操作后,效果和原来的插入排序是一样的,但是效率却提升了不止一点,下面我们还是来看代码:

#pragma  mark -  优化后的插入排序
#pragma  mark -
/**
 优化后的插入排序(将多次交换优化为赋值)
 
 @param array 未排序数组
 */
- (void)insertionSort2:(NSMutableArray *)array {
    for (int i = 0; i < array.count; i++) {
        NSNumber *temp = array[i];
        int j ;//这里的j要声明在外面 因为最后temp和前面的元素比较完后 j就是temp要存放的位置
        for ( j = i; j > 0 && array[j - 1] > temp ; j--) {//只有j大于0以及前一个元素比比当前temp这个元素要大才让元素后移
                array[j] = array[j-1];
        }
        //经历了上面循环后 temp就找到了合适的插入位置了
        array[j] = temp;
    }
}

到这里,我们的插入排序也就完成了,当然我知道你们肯定要说了,你说提升了效率,那到底提升了多少你没说啊,别急,下面我就来测试一下:

    NSMutableArray *array1 = [self.testHelper generateRandomArray:50000 rangeLeft:1 rangeRight:100];
   NSMutableArray * insertionSort1 = array1.mutableCopy;
    NSMutableArray * insertionSort2 = array1.mutableCopy;
    NSLog(@"=============未优化的插入排序耗时==============");
    [self.testHelper testSortWithExcuteBlock:^{
        [self insertionSort1:insertionSort1];
    }];

    NSLog(@"=============优化后的插入排序耗时==============");
    [self.testHelper testSortWithExcuteBlock:^{
        [self insertionSort2:insertionSort2];
    }];

这里我们同样以5万的数量级来进行测试,发现测试结果如下:

插入排序优化前后对比.png

是不是很惊讶?优化后的插入排序是不是明显比未优化的插入排序提升了不少,大家也快去试验一下吧。最后再给大家提供一个第三个版本的插入排序,也就是给出两个边界left和right来进行排序的代码:

#pragma  mark -  插入排序3 根据传入的左边界和右边界进行排序
#pragma  mark -
- (void)insertionSort3:(NSMutableArray *)array left:(int)left right:(int)right {
    for (int i = left +1; i <= right; i++) {
        NSNumber *tempNum = array[i];
        int j ;
        for ( j = i; j > left && array[j - 1] > tempNum; j--) {
            array[j] = array[j - 1];
        }
        array[j] = tempNum;
    }
}
Swift版本的插入排序

最近看swift,所以更新一波

        for i in 1..<array.count {
            for j in (1...i).reversed(){
                if array[j] < array[j-1] {
                    array.swapAt(j, j-1)
                }
            }
        }
        

好吧,插入排序就讲这么多了,毕竟插入排序并不属于高级排序,下一篇文章我们来讲讲归并排序与插入排序的结合使用,欢迎关注

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

推荐阅读更多精彩内容