iOS开发实用技巧—Objective-C中的各种遍历(迭代)方式

简介:

说明:

  1)该文简短介绍在iOS开发中遍历字典、数组和集合的几种常见方式。

  2)该文对应的代码可以在下面的地址获得:https://github.com/HanGangAndHanMeimei/Code

一、使用for循环

  要遍历字典、数组或者是集合,for循环是最简单也用的比较多的方法,示例如下: 

复制代码
 1 //普通的for循环遍历  2 -(void)iteratorWithFor
 3 {
 4 //////////处理数组//////////  5 NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
 6 NSInteger arrayMCount = [arrayM count];
 7 for (int i = 0; i<arrayMCount; i++) {
 8 NSString *obj = arrayM[i];
 9 NSLog(@"%@",obj);
10  }
11 12 //////////处理字典////////// 13 NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
14 NSArray *dictKeysArray = [dictM allKeys];
15 for (int i = 0; i<dictKeysArray.count; i++) {
16 NSString *key = dictKeysArray[i];
17 NSString *obj = [dictM objectForKey:key];
18 NSLog(@"%@:%@",key,obj);
19  }
20 21 //////////处理集合////////// 22 NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
23 NSArray *setObjArray = [setM allObjects];
24 for (int i = 0; i<setObjArray.count; i++) {
25 NSString *obj = setObjArray[i];
26 NSLog(@"%@",obj);
27  }
28 29 //////////反向遍历----降序遍历----以数组为例 30 NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
31 NSInteger arrayMCount2 = [arrayM2 count] - 1;
32 33 for (NSInteger i = arrayMCount2; i>0; i--) {
34 NSString *obj = arrayM2[i];
35 NSLog(@"%@",obj);
36  }
37 }
复制代码

优点:简单

缺点:由于字典和集合内部是无序的,导致我们在遍历字典和集合的时候需要借助一个新的『数组』作为中介来处理,多出了一部分开销。

二、使用NSEnumerator遍历

NSEnumerator的使用和基本的for循环类似,不过代码量要大一些。示例如下:

复制代码
 1 //使用NSEnumerator遍历  2 -(void)iteratorWithEnumerator
 3 {
 4 //////////处理数组//////////  5 NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
 6 NSEnumerator *arrayEnumerator = [arrayM objectEnumerator];
 7 NSString *obj;
 8 while ((obj = [arrayEnumerator nextObject]) != nil) {
 9 NSLog(@"%@",obj);
10  }
11 12 //////////处理字典////////// 13 NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
14 NSEnumerator *dictEnumerator = [dictM keyEnumerator];
15 NSString *key;
16 while ((key = [dictEnumerator nextObject]) != nil) {
17 NSString *obj = dictM[key];
18 NSLog(@"%@",obj);
19  }
20 21 22 //////////处理集合////////// 23 NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
24 NSEnumerator *setEnumerator = [setM objectEnumerator];
25 NSString *setObj;
26 while ((setObj = [setEnumerator nextObject]) != nil) {
27 NSLog(@"%@",setObj);
28  }
29 30 31 //////////反向遍历----降序遍历----以数组为例 32 NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
33 NSEnumerator *arrayEnumerator2 = [arrayM2 reverseObjectEnumerator];
34 NSString *obj2;
35 while ((obj2 = [arrayEnumerator2 nextObject]) != nil) {
36 NSLog(@"%@",obj2);
37  }
38 39 }
复制代码

优点:对于不同的数据类型,遍历的语法相似;内部可以简单的通过reverseObjectEnumerator设置进行反向遍历。

缺点:代码量稍大。

三、使用for...In遍历

在Objective-C 2.0 中增加了for ...In 形式的快速遍历。此种遍历方式语法简洁,速度飞快。示例如下:

复制代码
 1 //使用for...In进行快速遍历  2 -(void)iteratorWithForIn
 3 {
 4 //////////处理数组//////////  5 NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
 6 for (id obj in arrayM) {
 7 NSLog(@"%@",obj);
 8  }
 9 10 //////////处理字典////////// 11 NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
12 for (id obj in dictM) {
13 NSLog(@"%@",dictM[obj]);
14  }
15 16 //////////处理集合////////// 17 NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
18 for (id obj in setM) {
19 NSLog(@"%@",obj);
20  }
21 22 //////////反向遍历----降序遍历----以数组为例 23 NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
24 for (id obj in [arrayM2 reverseObjectEnumerator]) {
25 NSLog(@"%@",obj);
26  }
27 }
复制代码

优点:1)语法简洁;2)效率最高;

缺点:无法获得当前遍历操作所针对的下标。

四、基于Block的遍历方式

基于Block的方式来进行遍历是最新引入的方法。它提供了遍历数组|字典等类型数据的最佳实践。示例如下:

复制代码
 1 //基于块(block)的遍历方式  2 -(void)iteratorWithBlock
 3 {
 4 //////////处理数组//////////  5 NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
 6 [arrayM enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
 7 NSLog(@"%zd--%@",idx,obj);
 8  }];
 9 10 //////////处理字典////////// 11 NSDictionary *dictM = @{@"1":@"one",@"2":@"two",@"3":@"three"};
12 [dictM enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
13 NSLog(@"%@:%@",key,obj);
14  }];
15 16 //////////处理集合////////// 17 NSSet * setM = [[NSSet alloc] initWithObjects:@"one",@"two",@"three",@"four", nil];
18 [setM enumerateObjectsUsingBlock:^(id _Nonnull obj, BOOL * _Nonnull stop) {
19 NSLog(@"%@",obj);
20  }];
21 22 //////////反向遍历----降序遍历----以数组为例 23 NSArray *arrayM2 = @[@"1",@"2",@"3",@"4"];
24 [arrayM2 enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
25 NSLog(@"%zd--%@",idx,obj);
26  }];
27 }
复制代码

优点:1)遍历时可以直接从block中获得需要的所有信息,包括下标、值等。特别相对于字典而言,不需要做多余的编码即可同时获得key和value的值。

   2)能够直接修改block中key或者obj的类型为真实类型,可以省去类型转换的工作。

   3)可以通过NSEnumerationConcurrent枚举值开启并发迭代功能。

说明:基于Block的遍历方式在实现反向遍历的时候也非常简单,使用enumerateObjectsWithOptions方法,传递NSEnumerationReverse作为参数即可,在处理遍历操作的时候推荐基于Block的遍历方式。

五、使GCD中的dispatch_apply函数

  使用GCD中的dispatch_apply函数也能实现字典、数组等的遍历,该函数比较适合处理耗时较长、迭代次数较多的情况。示例如下:

复制代码
 1 //使用GCD中的dispatch_apply函数  2 -(void)iteratorWithApply
 3 {
 4 //////////处理数组//////////  5 NSArray *arrayM = @[@"1",@"2",@"3",@"4"];
 6  7 //获得全局并发队列  8 dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
 9 10 dispatch_apply(arrayM.count, queue, ^(size_t index) {
11 NSLog(@"%@--%@",arrayM[index],[NSThread currentThread]);
12  });
13 }
复制代码

优点:开启多条线程并发处理遍历任务,执行效率高。

缺点:1)对于字典和集合的处理需借助数组;2)无法实现反向遍历。

目录
相关文章
|
4天前
|
开发工具 Android开发 iOS开发
探索Android与iOS开发的差异与挑战
【7月更文挑战第11天】在移动应用开发的广阔天地中,Android和iOS两大平台如同双子星座般耀眼,各自拥有独特的开发生态和用户群体。本文将深入分析这两个平台的显著差异,从技术架构到开发工具,再到市场定位,揭示它们之间的异同。通过比较,我们不仅能够更好地理解各自的优势和局限,还能洞察未来移动应用开发的趋势。
|
2天前
|
前端开发 开发工具 Swift
学习iOS开发的准备
准备学习iOS开发?确保有Mac和最新Xcode,先学好编程基础特别是Swift。利用Apple官方文档、在线课程和书籍作为资源。熟悉Xcode及Git,通过实践项目和开源代码积累经验。深研架构模式、核心框架和优化技巧。加入开发者社区,关注行业动态,持续学习。
14 1
|
7天前
|
移动开发 开发工具 Android开发
探索安卓与iOS开发的差异:平台特性与编程实践
【7月更文挑战第8天】在移动开发的广阔天地中,安卓和iOS这两大操作系统各自占据着半壁江山。它们在用户界面设计、系统架构及开发工具上展现出截然不同的特色。本文将深入探讨这两个平台在技术实现和开发生态上的关键差异,并分享一些实用的开发技巧,旨在为跨平台开发者提供有价值的见解和建议。
|
4天前
|
Java 开发工具 Android开发
探索Android与iOS开发的差异与挑战
【7月更文挑战第11天】在移动应用开发的广阔天地中,Android和iOS两大平台各领风骚。本文将深入探讨这两个平台的开发差异,从编程语言、用户界面设计到开发工具等方面进行比较,并分析开发者面临的挑战。通过对比分析,旨在为开发者提供一个全面的视角,帮助他们更好地选择适合自己项目需求的开发平台。
8 0
|
4天前
|
iOS开发 开发者 UED
探索iOS开发中的SwiftUI框架
【7月更文挑战第11天】本文将深入探讨SwiftUI框架,这是苹果公司为iOS开发者提供的一个现代化、声明式的用户界面工具。我们将了解SwiftUI如何简化界面设计流程,提升代码的可读性和可维护性,并讨论其在实际应用中的优势和局限性。通过实际案例分析,本文旨在帮助读者更好地理解SwiftUI的核心概念和应用实践。
|
5天前
|
机器学习/深度学习 人工智能 iOS开发
探索iOS开发的未来:SwiftUI的革新之旅
在数字时代的浪潮中,iOS开发正经历一场由SwiftUI引领的变革。本文将深入探讨SwiftUI如何重塑移动应用开发,提升开发者的生产力,并展望其对iOS生态的影响。通过分析SwiftUI的核心特性、实际案例和未来趋势,我们将揭示这一现代框架如何定义用户体验的新标准,同时为iOS开发者指明技术进阶之路。
|
6天前
|
搜索推荐 Android开发 iOS开发
探索Android与iOS开发的差异:平台特性与用户体验的对比分析
【7月更文挑战第9天】在移动应用开发的浩瀚海洋中,Android和iOS两大操作系统如同两座灯塔,指引着开发者们的航向。本文将深入探讨这两个平台在开发环境、用户界面设计、性能优化以及市场策略上的根本差异。我们将通过比较分析,揭示各自平台的独特优势和潜在挑战,为开发者提供决策支持,同时也为用户体验的提升指明方向。
|
开发工具 数据安全/隐私保护 iOS开发
|
2月前
|
安全 编译器 Swift
IOS开发基础知识: 对比 Swift 和 Objective-C 的优缺点。
IOS开发基础知识: 对比 Swift 和 Objective-C 的优缺点。
169 2
|
9天前
|
开发工具 iOS开发 容器
【Azure Blob】关闭Blob 匿名访问,iOS Objective-C SDK连接Storage Account报错
iOS Objective-C 应用连接Azure Storage时,若不关闭账号的匿名访问,程序能正常运行。但关闭匿名访问后,上传到容器时会出现错误:“Public access is not permitted”。解决方法是将创建容器时的公共访问类型从`AZSContainerPublicAccessTypeContainer`改为`AZSContainerPublicAccessTypeOff`,以确保通过授权请求访问。
【Azure Blob】关闭Blob 匿名访问,iOS Objective-C SDK连接Storage Account报错