在iOS开发中,可以使用多种方法进行元素遍历,具体有一下几种:
经典for循环
1
2
3
4
5
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; for ( int i = 0; i < iosArray.count; i++) { //处理数组中数据 NSLog(@ "%@" , iosArray[i]); } |
NSEnumerator遍历
1
2
3
4
5
6
7
8
9
10
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; NSEnumerator *enumerator = [iosArray objectEnumerator]; //正向遍历 // NSEnumerator *enumerator = [iosArray reverseObjectEnumerator];//反向遍历 id object; while ((object = [enumerator nextObject]) != nil) { //处理枚举器中的数据 NSLog(@ "%@" , object); } |
for-in快速遍历
1
2
3
4
5
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; for (NSString *obj in iosArray) { //处理数组中的数据 NSLog(@ "%@" , obj); } |
EnumeratorBlock遍历
1
2
3
4
5
6
7
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; [iosArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) { NSLog(@ "%@" , obj); if ([obj isEqualToString:@ "e" ]) { *stop = YES; // 跳出遍历 } }]; |
另外,EnumeratorBlock还支持反向遍历,并发遍历,并发遍历可以使用多核的优化,充分利用系统的资源。
反向遍历
1
2
3
4
5
6
7
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; [iosArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) { NSLog(@ "%@" , obj); if ([obj isEqualToString:@ "e" ]) { *stop = YES; } }]; |
并发遍历
1
2
3
4
5
6
7
8
9
10
11
|
NSArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; NSMutableArray *iosMutableArray = [NSMutableArray arrayWithArray:iosArray]; [iosMutableArray enumerateObjectsWithOptions:NSEnumerationConcurrent usingBlock:^(NSString *obj, NSUInteger idx, BOOL * _Nonnull stop) { obj = [NSString stringWithFormat:@ "_%@" , obj]; [iosMutableArray replaceObjectAtIndex:idx withObject:obj]; NSLog(@ "%@" , obj); if ([obj isEqualToString:@ "_I" ]) { *stop = YES; } }]; |
dispatch_apply遍历
dispatch_apply类似于for循环,这里需要注意的是,dispatch_apple是同步调用,调用完毕返回结果,并且由于是GCD实现,所以可以使用并发队列或者是串行队列。代码如下:
1
2
3
4
5
6
|
dispatch_queue_t queue = dispatch_queue_create( "queue" , DISPATCH_QUEUE_CONCURRENT); // dispatch_queue_t queue = dispatch_queue_create("queue", DISPATCH_QUEUE_SERIAL); // 串行队列 dispatch_apply(array.count, queue, ^( size_t i) { Enumerate *enumerate = [array objectAtIndex:i]; NSLog(@ "number: %ld" , enumerate.number); }); |
遍历的注意事项
for循环中不要修改数组
遍历过程中是不能随便删除遍历的元素的,如果需要删除元素,可以先复制一份出来,比如如下的代码会有问题:
1
2
3
4
5
6
7
|
NSMutableArray *iosArray = @[@ "a" , @ "b" , @ "c" , @ "d" , @ "e" , @ "f" , @ "g" ]; for (NSString *obj in iosArray) { //处理数组中的数据 if ([@ "e" isEqualTo:obj]) { [iosArray removeObject:obj]; } } |
但是使用enumerateBlock可以在block内部做removeObject操作,原因应该是和Block的特性有关, 在Block中会保存变量的值,而不会随变量的值的改变而改变 。
遍历的速率
当数组容量很大的时候,如果只是进行数组遍历的话,使用for-in是最快速的,其次是并发遍历,这个很多人都以为enumerateBlock是最快的。
遍历实践tips
数组分组
在开发中,有时需要对数组进行某种情况的分组,比如,一个拥有很多消息模型的数组,我们需要根据消息的创建月份进行分组,那么可以使用下面的方法实现:
1
2
3
4
5
6
7
8
9
10
|
NSMutableSet *set=[NSMutableSet set]; NSArray *array = @[message1, message2, message3, message4, message5, message6, message7]; __block NSArray *tempDataArray = [NSArray arrayWithArray:array]; [tempDataArray enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) { [set addObject:obj.month]; //利用set不重复的特性,得到有多少组,根据数组中消息的月份属性 }]; [set enumerateObjectsUsingBlock:^(id obj, BOOL *stop) { //遍历set数组 NSPredicate *predicate = [NSPredicate predicateWithFormat:@ "SELF.month = %@" , obj]; //创建谓词筛选器 NSArray *group = [tempDataArray filteredArrayUsingPredicate:predicate]; //用数组的过滤方法得到新的数组,在添加的最终的数组 } |
倒序遍历
倒序遍历也很常见,可以使用上面的反向遍历来实现。
set排序
这个和Emunerate其实没有关系,但是也很实用,我们知道set是无序的,但是有时需要实现有顺序的set,可以使用下面来实现:
1
2
3
|
//由于set无序,现将set转换成nsarray NSArray *sortDescriptor = @[[[NSSortDescriptor alloc] initWithKey:@ "self" ascending:NO]]; NSArray *sortSetArray = [set sortedArrayUsingDescriptors:sortDescriptor]; |
其实原理是将set转化成array来实现的。
总结
遍历在我们日常开发中十分常见,根据应用场景,选择合适的遍历方法才是我们需要关系的。这篇文章总结了下遍历的种类和注意事项,希望能帮助到有需要的同学。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:http://www.tuicool.com/articles/IZFbae7