数组遍历是编码中很常见的一种需求,我们来扒一拔iOS里面都有什么样的方法来实现,有什么特点。

因为ios是兼容C语言的,所以c语言里面的最最常见的for循环遍历是没有问题的。

本文中用的数组是获取的系统的语言数组,大约有30多个数据,虽然还不够模拟大批量的数据,但对于方法的验证是没有问题的了。

  1. NSArray *langArray = [[NSUserDefaults standardUserDefaults] arrayForKey:@"AppleLanguages"];

第一种方法是最最熟悉的C语言演化过来的:

  1. for (int i = 0; i<langArray.count; i++) {
  2. NSLog(@"langArray[%d]=%@", i, langArray[i]);
  3. }

这个方法最普通,效率也一般,但它也有好处,一是方便针对下标的处理,就是说如果我要处理i==10的情况是很简便的,另一个是可以比较方便的反向遍历。

Objective-C 1.0里面的NSEnumerator也可以进行遍历,代码如下:

  1. NSEnumerator *enumerator = [langArray objectEnumerator];
  2. id object;
  3. while ((object = [enumerator nextObject]) != nil) {
  4. NSLog(@"langArray=%@", object);
  5. }

这里我们可以看到没有下标了,通过nextObject的方法来遍历。这个方法的好处是对于遍历NSDictionary和NSSet代码也比较类似,不便的是对于下标的处理会不方便,另外反向遍历需要用reverseObjectEnumerator方法。

objective-c发展到2.0时又有了快速遍历功能,代码如下:

  1. for (id object in langArray) {
  2. NSLog(@"langArray=%@", object);
  3. }

这里代码简洁清晰,很长时间是我写代码的首选,号称效率也最高,不过不便之处同样明显,如果算法要求知道数组的下标,这个方法就抓瞎了。另外,反向需要通过[langArray reverseObjectEnumerator]来实现。

等到block出来后,iOS里面新增加了enumerateObjectsUsingBlock:的方法,代码如下:

  1. [langArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOLBOOL *stop) {
  2. NSLog(@"idx=%d, id=%@", idx, obj);
  3. }];

这里我们看到block里面的参数包括object,下标以及是否停止遍历,应该说,这个能满足基本所有的遍历需求了,有下标,有运行的对象,还有是否继续遍历的标志。不过反向遍历呢?苹果提供了另外一个方法:

  1. [langArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id obj, NSUInteger idx, BOOLBOOL *stop) {
  2. NSLog(@"idx=%d, id=%@", idx, obj);
  3. }];

这个enumerateObjectsWithOptions:usingBlock:方法比enumerateObjectsUsingBlock:方法多传了一个参数,这个参数指定了遍历的顺序。

说到这里,如果我们选择正向遍历,那么这两种方法是一样的么?答案也是否定的。在enumerateObjectsWithOptions:usingBlock:方法里面,如果指定了NSEnumerationConcurrent顺序,那么底层通过GCD来处理并发执行事宜,具体实现可能会用到dispatch group。也就是说,这个会用多线程来并发实现,并不保证按照顺序执行,但效率肯定是杠杠的!

我们来看一下打印结果:

  1. 2014-06-17 15:46:44.413 testStoryboard[2703:3503] idx=32, id=hu
  2. 2014-06-17 15:46:44.413 testStoryboard[2703:1303] idx=16, id=ru
  3. 2014-06-17 15:46:44.416 testStoryboard[2703:3503] idx=33, id=vi
  4. 2014-06-17 15:46:44.412 testStoryboard[2703:60b] idx=0, id=zh-Hant
  5. 2014-06-17 15:46:44.417 testStoryboard[2703:1303] idx=17, id=pl
  6. 2014-06-17 15:46:44.417 testStoryboard[2703:60b] idx=1, id=zh-Hans
  7. 2014-06-17 15:46:44.417 testStoryboard[2703:1303] idx=18, id=tr
  8. 2014-06-17 15:46:44.419 testStoryboard[2703:60b] idx=2, id=en
  9. 2014-06-17 15:46:44.419 testStoryboard[2703:1303] idx=19, id=uk
  10. 2014-06-17 15:46:44.421 testStoryboard[2703:60b] idx=3, id=fr
  11. 2014-06-17 15:46:44.421 testStoryboard[2703:1303] idx=20, id=ar
  12. 2014-06-17 15:46:44.421 testStoryboard[2703:60b] idx=4, id=de
  13. 2014-06-17 15:46:44.422 testStoryboard[2703:60b] idx=5, id=ja
  14. 2014-06-17 15:46:44.422 testStoryboard[2703:60b] idx=6, id=nl
  15. 2014-06-17 15:46:44.421 testStoryboard[2703:1303] idx=21, id=hr
  16. 2014-06-17 15:46:44.423 testStoryboard[2703:60b] idx=7, id=it
  17. 2014-06-17 15:46:44.423 testStoryboard[2703:1303] idx=22, id=cs
  18. 2014-06-17 15:46:44.423 testStoryboard[2703:60b] idx=8, id=es
  19. 2014-06-17 15:46:44.424 testStoryboard[2703:1303] idx=23, id=el
  20. 2014-06-17 15:46:44.424 testStoryboard[2703:60b] idx=9, id=ko
  21. 2014-06-17 15:46:44.424 testStoryboard[2703:1303] idx=24, id=he
  22. 2014-06-17 15:46:44.425 testStoryboard[2703:60b] idx=10, id=pt
  23. 2014-06-17 15:46:44.425 testStoryboard[2703:60b] idx=11, id=pt-PT
  24. 2014-06-17 15:46:44.425 testStoryboard[2703:1303] idx=25, id=ro
  25. 2014-06-17 15:46:44.426 testStoryboard[2703:60b] idx=12, id=da
  26. 2014-06-17 15:46:44.426 testStoryboard[2703:1303] idx=26, id=sk
  27. 2014-06-17 15:46:44.426 testStoryboard[2703:60b] idx=13, id=fi
  28. 2014-06-17 15:46:44.426 testStoryboard[2703:1303] idx=27, id=th
  29. 2014-06-17 15:46:44.427 testStoryboard[2703:60b] idx=14, id=nb
  30. 2014-06-17 15:46:44.427 testStoryboard[2703:1303] idx=28, id=id
  31. 2014-06-17 15:46:44.428 testStoryboard[2703:60b] idx=15, id=sv
  32. 2014-06-17 15:46:44.428 testStoryboard[2703:1303] idx=29, id=ms
  33. 2014-06-17 15:46:44.429 testStoryboard[2703:1303] idx=30, id=en-GB
  34. 2014-06-17 15:46:44.429 testStoryboard[2703:1303] idx=31, id=ca

从这个结果我们可以看出,确实遍历了整个数组,但并发按照顺序从头到尾——也就是说,用到了dispatch group。这在遍历大数组而有相互独立时对于效率的提高是相当有利的,赞一个!

在iOS中,除数组外,还有NSDictionary和NSSet数据也是称为collection数据的,遍历有类似的地方,不过遍历没有数组那么频繁,方法上是差不多的。

 

iOS中数组遍历的方法及比较的更多相关文章

  1. iOS中数组遍历的方法及比較

    数组遍历是编码中非经常见的一种需求.我们来扒一拔iOS里面都有什么样的方法来实现,有什么特点. 由于iOS是兼容C语言的.所以C语言里面的最最常见的for循环遍历是没有问题的. 本文中用的数组是获取的 ...

  2. iOS中产生随机数的方法

    利用arc4random_uniform()产生随机数 Objective-C 中有个arc4random()函数用来生成随机数且不需要种子,但是这个函数生成的随机数范围比较大,需要用取模的算法对随机 ...

  3. ios中NSUserDefaults的使用方法

    ios中NSUserDefaults的使用方法 NSUserDefaults类提供了一个与默认系统进行交互的编程接口.NSUserDefaults对象是用来保存.恢复应用程序相关的偏好设置,配置数据等 ...

  4. js中数组遍历常用的方法

    常见的数组遍历方法,比如 for in,for  of, forEach,map,filter,every,some,find,reduce等 1,普通for循环,经常用的数组遍历 var arr = ...

  5. javascript中数组常用的方法

    在JavaScript中,数组可以使用Array构造函数来创建,或使用[]快速创建,这也是首选的方法.数组是继承自Object的原型,并且他对typeof没有特殊的返回值,他只返回'object'. ...

  6. javascript中数组Array的方法

    一.常用方法(push,pop,unshift,shift,join)push pop栈方法,后进先出var a =[1,2,3];console.log(a.push(40)); //4 返回数组的 ...

  7. ES6中数组的新方法

    数组的扩展 1.1扩展运算符 1.1.1:... 扩展运算符(spread)是三个点(...).它好比 rest 参数的逆运算,将一个数组转为用逗号分隔的参数序列. <body> < ...

  8. javascript中数组常用的方法和属性

    前言 在javascript中,数组是一种非常重要的数据类型,我们时常会和它打交道,最近在开发项目中频繁的使用到数组,但是自己对数组的众多方法已经是非常模糊了,为了方便自己以后能够更好的使用数组中的属 ...

  9. JS中数组和字符串方法的简单整理

    一.数组: 数组的基本方法:              1.增:arr.unshift() /push()    前增/后增                  2.删:arr.shift() /pop ...

随机推荐

  1. kylin入门到实战:cube详述

    版权申明:转载请注明出处. 文章来源:http://bigdataer.net/?p=306 排版乱?请移步原文获得更好的阅读体验 1.什么是cube? cube是所有dimession的组合,每一种 ...

  2. [PyTorch]PyTorch中模型的参数初始化的几种方法(转)

    ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ 本文目录 1. xavier初始化 2. kaiming初始化 3. 实际使用中看到的初始化 3.1 ResNeXt,de ...

  3. install opencv

    OpenCV是一个基于开源发行的跨平台计算机视觉库,它轻量级而且高效——由一系列 C 函数和少量 C++ 类构成,同时提供了Python.Ruby.MATLAB等语言的接口,实现了图像处理和计算机视觉 ...

  4. Objective C Protocol implementation

    protocol 类似于接口,可以实现函数的回调 @protocol MyDelegate<NSObject> -(void)myCallbackFunction; @end //Call ...

  5. sleep(),wait(),yield()和join()方法的区别

    sleep() sleep()方法需要指定等待的时间,它可以让当前正在执行的线程在指定的时间内暂停执行,进入阻塞状态,该方法既可以让其他同优先级或者高优先级 的线程得到执行的机会,也可以让低优先级的线 ...

  6. java高级特性(1)--理解面向对象思想

    前言: 优秀的代码具备:高性能.可重用.可扩展.易维护.易理解 具体实现: 高性能:合理使用算法,数据结构等等 可重用:封装.继承 可扩展:多态 易维护.易理解:命名规范 + 注解 面向对象是一种思想 ...

  7. puma(5300✨) Rails的一个多线程,高并发处理的web server

    https://github.com/puma/puma 在开发和产品环境下,Puma是一个简单,快速,多线程,并高并发highly concurrent HTTP1.1server for Ruby ...

  8. 解决Tomcat加载时报APR错的问题

    部署Tomcat的时候出现了如下错误, INFO: The APR based Apache Tomcat Native library which allows optimal performanc ...

  9. 转载:【Oracle 集群】RAC知识图文详细教程(五)--特殊问题和实战经验

    文章导航 集群概念介绍(一) ORACLE集群概念和原理(二) RAC 工作原理和相关组件(三) 缓存融合技术(四) RAC 特殊问题和实战经验(五) ORACLE 11 G版本2 RAC在LINUX ...

  10. Prism 4 文档 ---第8章 导航

        作为同用户具有丰富的交互的客户端应用程序,它的用户界面(UI)将会持续不断的更新来反映用户工作的当前的任务和数据.用户界面可以进行一段时间相当大的变化作为用户交互的应用程序中完成各种任务.通过 ...