一直以为for循环和 for in 是一样的,例如:
CFAbsoluteTime start = CFAbsoluteTimeGetCurrent();
NSMutableArray * arr = @[@1,@2,@3,@4,@5].mutableCopy;
for (NSNumber *obj in arr) {
NSLog(@"%@",obj);
}
CFAbsoluteTime end = CFAbsoluteTimeGetCurrent();
NSLog(@"for in cost: %0.3f",end - start);
CFAbsoluteTime startTime = CFAbsoluteTimeGetCurrent();
for (int i = 0; i < arr.count; i ++) {
NSLog(@"%@",arr[i]);
}
CFAbsoluteTime endTime = CFAbsoluteTimeGetCurrent();
NSLog(@"for cost: %0.3f",endTime - startTime);
但是偶然间写了这么一段就崩溃了
for (NSString * obj in arr) {
if ([obj integerValue] == 2) {
[arr addObject:@"6"];
}
}
CFAbsoluteTime end = CFAbsoluteTimeGetCurrent();
NSLog(@"for in cost: %0.3f Arr:%@",end - start,arr);
但是如果写成for循环就没有问题,然后就去百度了一下发现:如果在for in 循环里,对这个数组进行了修改的话,无论是增,删,修改数组元素位置,都会扔一个异常出来,枚举的过程中数组发生了突变(<__NSArrayM: 0xa4fc000> was mutated while being enumerated.)
for in实际上是快速枚举,跟for循环意义上还是有区别的。NSArray的枚举操作中有一条需要注意:对于可变数组进行枚举操作时,不能通过添加或删除对象等这类操作来改变数组容器,否则就会报错.而本身这种操作也是有问题的,数组容器已经改变,可能遍历到没有分配的位置,用for循环机器不能自己察觉,但是枚举器可以察觉。
enumerateKeysAndObjectsUsingBlock用法
NSDictionary * dic = [NSDictionary dictionaryWithObjectsAndKeys:@"obj1",@"key1",@"obj2",@"key2", nil];
[dic enumerateKeysAndObjectsUsingBlock:^(id key, id value, BOOL *stop) {
NSLog(@"value for key %@ is %@ ", key, value);
if ([@"key2" isEqualToString:key]) {
*stop = YES;
}
}];
NSDictionary有一个方法叫enumerateKeysAndObjectsUsingBlock,它就一个参数就是block,这个block携带了三个参数,这将要把dictionary里面的key和value每次一组传递到block,enumerateKeysAndObjectsUsingBlock会遍历dictionary并把里面所有的key和value一组一组的展示给你,每组都会执行这个block。这其实就是传递一个block到另一个方法,在这个例子里它会带着特定参数被反复调用,直到找到一个key2的key,然后就会通过重新赋值那个BOOL *stop来停止运行,停止遍历同时停止调用block。
更详细的用法
(http://blog.itpub.net/12231606/viewspace-1084119/)
enumerateObjectsWithOptions使用
NSArray *array = [[NSArray alloc]initWithObjects:@"a",@"b",@"c",@"d",@"e",@"f", nil];
//遍历数组元素
[array enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"obj=%@ idx=%ld",obj,idx);
}];
//如果指定了NSEnumerationConcurrent顺序,那么底层通过GCD来处理并发执行事宜,具体实现可能会用到dispatch group。也就是说,这个会用多线程来并发实现,并不保证按照顺序执行
//NSEnumerationReverse 倒序排列
[array enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"idx=%ld, id=%@", idx, obj);
//当需要结束循环的时候,调用stop,赋予YES
if (idx ==3) {
*stop = YES;
}
}];
//NSIndexSet类代表一个不可变的独特的无符号整数的集合,称为索引,因为使用它们的方式。这个集合被称为索引集 唯一的,有序的,无符号整数的集合
[NSIndexSet indexSetWithIndex:1];//创建一个索引集合,根据索引值
[NSIndexSet indexSetWithIndexesInRange:NSMakeRange(0,8)];//创建一个索引集合,根据一个NSRange对象
[array enumerateObjectsAtIndexes:[NSIndexSet indexSetWithIndexesInRange:NSMakeRange(0,3)] options:NSEnumerationReverse usingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSLog(@"\n\n\nidx=%ld, id=%@", idx, obj);
}];
for in、经典for循环和EnumerateObjectsUsingBlock 的比较
- 对于集合中对象数很多的情况下,for in 的遍历速度非常之快,但小规模的遍历并不明显(还没普通for循环快)
- Value查询index的时候, 面对大量的数组推荐使用
enumerateObjectsWithOptions
的并行方法. - 遍历字典类型的时候, 推荐使用
enumerateKeysAndObjectsUsingBlock
,block版本的字典遍历可以同时取key和value(forin只能取key再手动取value)
参考文章:
(http://www.jianshu.com/p/ef3f1731a353)
(http://blog.sunnyxx.com/2014/04/30/ios_iterator/)