数组遍历是编码中很常见的一种需求,我们来扒一扒iOS里面都有什么样的方法来实现,有什么特点。
因为iOS是兼容C语言的,所以C语言里面的最最常见的for循环遍历是没有问题的。
NSArray * langArray = @[@"张三",@"李四",@"王五",@"马六"];
#pragma mark -- 第一种方法是最最熟悉的C语言演化过来的:
for (int i = 0; i<[langArray count]; i++) {
NSLog(@"langArr[%d] = %@",i,langArray[i]);
}
#pragma mark -- Objective-C 里面的NSEnumerator也可以进行遍历:
NSEnumerator *enumerator = [langArray objectEnumerator];
id object;
while ((object = [enumerator nextObject]) != nil) {
NSLog(@"langArr---\n(%@)",object);
}
这里我们可以看到没有下标了,通过nextObject的方法来遍历。这个方法的好处是对于 遍历NSDictionary和NSSet代码也比较类似,不便的是对于下标的处理会不方便,另外反向遍历需要用reverseObjectEnumerator方法。
#pragma mark --Objective-C发展到2.0时又有了快速遍历功能
for (id object in langArray) {
NSLog(@"langArr---%@",object);
}
这里代码简洁清晰,很长时间是我写代码的首选,号称效率也最高,不过不便之处同样明显,如果算法要求知道数组的下标,这个方法就抓瞎了。另外,反向需要通过[langArray reverseObjectEnumerator]来实现。
#pragma mark --等到block出来后,新增加了enumerateObjectsUsingBlock:的方法:
//1.第一种enumerate
[langArray enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"idx==%ld,obj=%@",idx,obj);
}];
//2.第二种enumerate,多了个NSEnumerationOptions参数,这个参数指定了遍历的顺序。
[langArray enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@" \n idx==%ld,obj=%@",idx,obj);
}];
[langArray enumerateObjectsWithOptions:NSEnumerationConcurrent usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@" \n idx==%ld,obj=%@",idx,obj);
}];
看到这里,如果我们选择正向遍历,那么这两种方法是一样的么?答案也是否定的。在enumerateObjectsWithOptions:usingBlock:方法里面,如果指定了NSEnumerationConcurrent顺序,那么底层通过GCD来处理并发执行事宜,具体实现可能会用到dispatch group。也就是说,这个会用多线程来并发实现,并不保证按照顺序执行,但效率肯定是杠杠的!
在iOS中,除数组外,还有NSDictionary和NSSet数据也是称为collection数据的,遍历有类似的地方,不过遍历没有数组那么频繁,方法上是差不多的。