学习IOS的一些简单的排序算法

开篇:
以下这是一些个人观点:
现在前端工作不断的更新发展,改朝换代,人们往往一直追随着这个框架那个框架的学习,慢慢的被 facekbook 、google 等公司牵着鼻子走,总感觉是没有静下心来分析的时间了,但是对于程序员来说,什么才是根本呢?所谓程序员根本,首先你得让你自己成为一个程序员,而不是打上标签 xx程序员,因为这只会令你步伐停止不前,只能观其表面,然后给这些所谓的大公司新技术给慢慢玩死。
慢慢地做开发也有些时候了,做得越久就是越有一种感觉,就是回归本质,但是我们常说的本质是什么呢?个人认为本质既为基础,而基础分为:数据库,算法,数据结构,编译原理,网络原理。所以作为一个有情怀的程序员,还是必须好好补一下这方面的知识。

那么,这个文集,我们就说一说数据结构和算法,究竟什么是数据结构呢? 数据结构是算法的副产物,它只是让我们更好的明白算法的本质,和更好的够造出更高性能的算法,以下,我们就先来说一说三个基础算法。

1 、 选择排序算法
插入排序的理解: 首先,找到数组中最小的那个元素 ,其次,将它和数组的第一个元素位置 (如果第一个元 就是最小元 那么它就和自己 )。再次,在剩下的元素中找到最小的元素, 将它与数组的第二个元素交换位置。如此反复, 到将整个数组排序。这种方法叫做做选择排序,因为它在不断地选择剩余元素之中的最小者。
解析:对于长度为 N 的数组,选择排序需要大约 N^2/2 次比较和 N 次交换。

图解:


选择排序图解

声明:

 NSMutableArray *selectSort(NSMutableArray *array, int start)

实现:

NSMutableArray *selectSort(NSMutableArray *array, int start) {

    if (start == array.count ) {
        return array;
    }
    
    int minNum = [array[start] intValue];
    int minIndex = start;
    for (int i = start ; i < array.count; i ++) {
        if (minNum > [array[i] intValue]) {
            minNum = [array[i] intValue];
            minIndex = i;
        }
    }
    
    [array exchangeObjectAtIndex:start withObjectAtIndex:minIndex];
    array = selectSort(array, start + 1);
    return array;
}

调用:

    NSArray *array = @[@(1), @(4), @(8), @(9), @(5), @(7), @(2)];
    array = selectSort([array mutableCopy], 0);

输出过程:

第一次:1, 4, 8, 9, 5, 7, 2
第二次:1, 2, 8, 9, 5, 7, 4
第三次:1, 2, 4, 9, 5, 7, 8
第四次:1, 2, 4, 5, 9, 7, 8
第五次:1, 2, 4, 5, 7, 9, 8
最后输出: 1, 2, 4, 5, 7, 8, 9

选择排序优势:稳定,它的比较次数基本上不会改变,数据移动比较少。

选择排序劣势:操作级别了指数级,不会因是否有序数组而改变排序时间,时间只会与数组长度有关系。


2、插入排序
插入排序的理解:由数组的第2个位置开始比较,若果前方位置的元素比较大,则交换位置,若自己元素较大,而继续下一个元素,如此排列,那么被操作的那个元素前方位置的所有元素皆为有序。
解析:对于随机排列的长度为 N 且主键不重复的数组,平均情况下插入排序需要~ N^2/4 次比较以及~ N^2/4 次交换。最坏情况下需要~ N^2/2 次比较和~ N^2/2 次交换,最好情况下需要 N-1次比较和 0 次交换。
插入排序需要的交换操作和数组中倒置的数量相同,需要的比较次数大于等于倒置的数量,小于等于倒置的数量加上数组的大小再减一。

图解:

屏幕快照 2016-03-18 上午1.24.56.png

声明:

NSMutableArray *InsetSort(NSMutableArray *mArray, NSInteger start) 

实现:

NSMutableArray *InsetSort(NSMutableArray *mArray, NSInteger start) {
    
    if (start == mArray.count) {
        return mArray;
    }
    
    for (NSInteger i = start; i > 0; i --) {
        if (mArray[i] < mArray[i-1]) {
            int temp = [mArray[i] intValue];
            int k =  (int)(i - 1);
            
            while (k >= 0 && [mArray[k] intValue] > temp) {
                mArray[k + 1] = mArray[k];
                k -= 1;
            }
            mArray[k+1] = @(temp);
        }
    }
    
    InsetSort(mArray, start + 1);
    
    return mArray;
}

调用:

NSMutableArray *mArray = [@[@(49),   @(38),   @(65),   @(97),   @(26),   @(13),   @(27),   @(49),   @(55),   @(4)] mutableCopy];
mArray = InsetSort(mArray, 1);

输出过程:

第一次:38,49,65,97,26,13,27,49,55,4
第二次:38,49,65,97,26,13,27,49,55,4
第三次:38,49,65,97,26,13,27,49,55,4
第四次:26,38,49,65,97,13,27,49,55,4
第五次:13,26,38,49,65,97,27,49,55,4
第六次:13,26,27,38,49,65,97,49,55,4
第七次:13,26,27,38,49,49,65,97,55,4
第八次:13,26,27,38,49,49,55,65,97,4
最后输出:4, 13, 26, 27, 38, 49,49, 55, 65, 97

插入排序优势:对于有序数组或部分有序数组,此排序方法是十分高效的,很适合小规模的数组,很多高级的排序算法都会利用到插入排序。

插入排序劣势:若果最少的元素都在最后部分的位置,那么该排序方法就会变得非常费劲了,最后的元素都要比较改元素位置减一次。


3、希尔排序
希尔排序为什么产生:希尔排序是以插入排序为基础的一种快速的排序算法。因为在大规模乱序数组中使用插入排序很慢,因为它只会交换相邻的两个元素,因此,如果越小的元素越是靠后,那么操作的复杂度将会大大提升,所以,人们把插入排序进行了改良,变成了希尔排序。
希尔排序的思想:希尔排序的思想是使数组中任意间隔为 h 的元素都是有序的。这样的数组为 h 有序数组。换句话说,一个 h 有序数组就是 h 个互相独 的有序数组编织在一起 成的一个数组。在进行排序时,如果 h 很大,我们就能将元素动到很远的地方,为实现更小的 h 有序创造方便。用这种方式,对于任意以 1 结尾的 h 序列,我们都能够将数组排序。这就是希尔排序。

图解:


希尔排序图解

前方高能。。。最后一个算法,用的C语言。

声明:

    void ShellSort(Array array)
void exchange(Array array, int N, int M) {
    int temp = array->a[N];
    array->a[N] = array->a[M];
    array->a[M] = temp;
}

struct ArrayNode {
    int a[kMaxLimit];
    int count;
};

void pushNumInArray(Array array, int Num) {
    if (array->count > 10) {
        printf("数组越界了");
        return;
    }
    
    array->a[array->count] = Num;
    array->count += 1;
}

void ShellSort(Array array) {
    for (int gap = array->count / 2; gap > 0; gap/=2) {
        for (int i = 0; i < gap; i ++) {
            
            if (array->a[i + gap] < array->a[i]) {
                for (int j = i + gap; j < array->count; j += gap) {
                    if (array->a[j] < array->a[j - gap]) {
                        // 交换
                        exchange(array, j, j-gap);
                    }
                }
            }
            
        }
    }
}

调用:

   Array array = malloc(sizeof(struct ArrayNode));
   array->count = 0;

for (int i = 0; i < 10; i ++) {
    pushNumInArray(array, arc4random() % 100);
 }
  
  // 希尔排序
  ShellSort(array);

  for (int i = 0; i < array->count; i ++) {
     printf("%d\n",array->a[i]);
 }

输出过程:

原数组: 49   38   65   97   26   13   27   49   55   4
第一次输出:gap = 5 / 2 = 2,数组为: 13   27   49   55   4    49   38   65   97   26
第二次输出:gap = 2 / 2 = 1,数组为:4   26   13   27   38    49   49   55   97   65
第三次输出:gap = 1 / 2 = 0,数组为:4   13   26   27   38    49   49   55   65   97

希尔排序优势:希尔排序优化了插入排序,在性能上比选择排序和插入排序快得多,而这种优势会随着数组越大变得越为明显。而且算法代码短简单,非常容易实现,所以我们基本上所有排序工作一开始都是用希尔排序,若在实际中不够快,我们再改成快速排序等更为高级的算法。

希尔排序劣势: 排序时间复杂度的下界是n*log2n。希尔排序没有快速排序算法快 O(n(logn)),因此中等大小规模表现良好,对规模非常大的数据排序不是最优选择。

好了,基础排序的三种算法就为以上三种了,希望大家都能看懂。


@end

心如止水,奋力前行

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

推荐阅读更多精彩内容

  • 概述:排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    每天刷两次牙阅读 3,726评论 0 15
  • 概述 排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部...
    蚁前阅读 5,164评论 0 52
  • 1.插入排序—直接插入排序(Straight Insertion Sort) 基本思想: 将一个记录插入到已排序好...
    依依玖玥阅读 1,235评论 0 2
  • 概述排序有内部排序和外部排序,内部排序是数据记录在内存中进行排序,而外部排序是因排序的数据很大,一次不能容纳全部的...
    Luc_阅读 2,253评论 0 35
  • 真的喝吐了… 大概两瓶半烧酒 要戒酒 好难受…
    snownow晓雪阅读 223评论 0 0