OC语言 - for | NSEnumerator | for...In | Block | GCD
for 循环
1 - 如果遍历字典、数组或者是集合,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 }
2 - 好坏
① 简单易懂
② 但由于字典、集合内部是无序的,导致在遍历它们的时候需要借助一个新的数组作为中介来处理,这就多出了一部分开销
NSEnumerator
1 - 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 }
2 - 好坏
① 对于不同的数据类型,其遍历的语法相似,内部可以简单地通过 reverseObjectEnumerator 设置进行反向遍历
② 代码量稍大
for...in
1 - 在 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 }
2 - 好坏
① 语法简洁、效率最高
② 无法获得当前遍历操作所针对的下标
基于 block 的遍历
1 - 它是最新引入的方法,提供了遍历数组字典等类型数据的最佳使用方式
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 }
2 - 好坏
① 遍历时可以直接从 block 中获得需要的所有信息,包括下标、值等。特别相对于字典而言,不需要做多余的编码即可同时获得 key 和 value 的值
② 能够直接修改 block 中 key 或者 obj 的类型为真实类型,可以省去类型转换的工作
③ 可以通过 NSEnumerationConcurrent 枚举值开启并发迭代功能
④ 它在实现反向遍历的时候也非常简单:使用 enumerateObjectsWithOptions 方法,传递 NSEnumerationReverse 作为参数即可
使用 dispatch_apply 函数
1 - 使用 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 }
2 - 好坏
① 开启多条线程并发处理遍历任务,执行效率高
② 对于字典和集合的处理需借助数组且无法实现反向遍历