原文网址:http://www.jianshu.com/p/ad80d9443a92
支持原创,如需转载, 请注明
你是不是以为你真的懂For...in... ??哈哈哈哈, 我也碰到了这个报错 .究其原因, 顾名思义 "在枚举的时候发生了变化"for...in...利用了快速枚举NSFastEnumerate当我们想要改变数组变量中的数据或者删除数组中的数据的时候,不能用for...in...应该是Objective-C中的foreach循环与Java中的相似,在内部是用iterator(迭代器)实现遍历的。而不管是在Java还是C++中,一旦修改了被遍历对象,在修改前生成的iterator都会失效,所以《C++ Primer》及Java课本中曾警告过不要在用iterator遍历集合时增删集合元素,看来Objective-C中也是一样。
NSArray的枚举操作中有一条需要注意:对于可变数组进行枚举操作时,你不能通过添加或删除对象这类操作来改变数组容器。如果你这么做了,枚举器会很困惑,而你将得到未定义的结果。所以我这里给出三种解决方案
1.利用for loop ()for (int i = 0; i < arr.count; i++) { if (...) { // do sth ; } }
2.利用临时变量
NSArray *tmp = [NSArray arrayWithArray:arr]; for (id obj in tmp) { if (...) { // do sth ; } }
3.enumerateObjectsUsingBlock(推荐)
[arr enumerateObjectsUsingBlock:^(NSMutableDictionary *obj, NSUInteger idx, BOOL *stop) { if (...) { // do sth *stop = YES; // 相当于break ; stop控制跳出枚举器. } }];
*** First throw call stack:
(0x353d088f 0x37777259 0x353d03b3 0x34e576a9 0x34ea914d 0x1252d9 0x16c281 0x16aadb 0x3425f887 0x3425f281 0x3425f911 0x16a931 0x3425dc59 0x34260817 0x3278ddfb 0x3278dcd0)
terminate called throwing an exception
*** First throw call stack:
(0x353d088f 0x37777259 0x353d03b3 0x34e576a9 0x34ea914d 0x1252d9 0x16c281 0x16aadb 0x3425f887 0x3425f281 0x3425f911 0x16a931 0x3425dc59 0x34260817 0x3278ddfb 0x3278dcd0)
terminate called throwing an exception2013-04-08 15:03:06.800 NanGuangTV[45122:707] *** Terminating app due to uncaught exception 'NSGenericException', reason: '*** Collection <__NSArrayM: 0x8ef350> was mutated while being enumerated.'
*** First throw call stack:
(0x353d088f 0x37777259 0x353d03b3 0x34e576a9 0x34ea914d 0x1252d9 0x16c281 0x16aadb 0x3425f887 0x3425f281 0x3425f911 0x16a931 0x3425dc59 0x34260817 0x3278ddfb 0x3278dcd0)
terminate called throwing an exception(lldb)
遇到这么个问题。郁闷。
原来是,我用了多线程,分别读取一个NSMutaleArray时报出来的Error!
原来,在同一时间,不同的线程同时读取和修改了NSMutaleArray。
根据这个提示, 想了一下,觉得应该在做遍历的过程中,修改了这个NSArray或者是NSMutableArray。后来想想, 这可能是多线程并发访问时出现的问题, 沿崩溃时的堆栈找了一下,果然找到。
因为我们在下载时, 使用了Block每当接受到数据,就去更新一下一个UIScrollView中的下载进度条的百分比。
而另有一个NSTimer, 会隔一段时间把这个UIScrollView中的全部元素清空,然后再重新添加, 因为可能有动态的元素需要添加或者删除。
运行时, 就会出现,两个线程同时去访问的情况, 当Block正在去更新UIScrollView中的这个元素时, NSTimer中可能正在执行清除操作,所以就崩了。
原来,在同一时间,不同的线程同时读取和修改了NSMutaleArray。
解决办法:避免多线程同时去修改一个对象, 避免做遍历时,这个对象被修改,根据这个思想,可以考虑加锁,或者直接使用atomic的方式来做。
第二种方法,利用变量, 你的意思是遍历的是 tmp 操作的是 arr 对吧.我有个疑问就是在操作完 arr 之后赋值给 tmp 这个不会造成 was mutated while being enumerate 这个错误么?
楼主,建议你应该写详细点,方法二补充下:
NSMutableArray *duplicateArray = [OriginalArray mutableCopy]; for(id obj in duplicateArray){ if ([ob isEqualToString:@"3"]){ [OriginalArray removeObject:obj]; } }