for循环删除数组中的元素crash问题

转载请注明出处!!!

如以下代码:

NSMutableArray *array = [NSMutableArray arrayWithObjects:@"2",@"3",@"4",@"9",@"4",@"12",@"22",@"4",@"4",@"5",@"6",@"1", nil];

     for (NSString *str in array) {

        if ([str isEqualToString:@"4"]) {

            [array removeObject:str];

        }

     }

     运行后肯定会crash,原因是数组越界。这是为什么?当我们正序快速遍历时,如果删除了一个,那么没有遍历到的元素位置都会往前移动一位,这样系统就无法确定接下来遍历是从删除位置开始呢,还是从删除位置下一位开始呢?这样就造成程序crash了。但是我们想要遍历删除时要怎么做?解决这个问题的方法有好几种。

     出现这个崩溃的事件只会出现在快速遍历中。如果像以下代码不采用快速遍历,而是使用for循环,就不会出现问题。

for (int i = 0; i < array.count; i++) {

        NSString *str  = array[i];

        if ([str isEqualToString:@"4"]) {

            [array removeObject:str];

        }

     }

     这两种有什么区别呢?

     https://developer.apple.com/library/ios/documentation/Cocoa/Reference/Foundation/Classes/NSEnumerator_Class/index.html

     这是Apple官方文档中关于快速遍历的利弊。其中的note中显示,建议最好不要再快速遍历的时候修改enumerator,否则不保证是安全的.由此就明白了,可能是我们在快速遍历的时候,移除掉一个元素,但是计数器依旧是原来的,那么在遍历到最后会继续调用nextObject方法,而此时实际上已经全部遍历完了,但是系统并不知道,还在遍历,也就是越界;当发现没有元素时,就crash了;而我写的常规遍历法为什么就可以呢,那是因为i < self.btnArray.count,这个判断条件中,当数组元素个数变化时,self.btnArray.count也在变,就不会出现数组越界的情况,因此第二种方法是可行的;

     第二种解决方法是定义一个副数组,遍历副本中的元素,在原数组中删除。

NSMutableArray *copyArray = [NSMutableArray arrayWithArray:array];

     for (NSString *str in copyArray) {

        if ([str isEqualToString:@"4"]) {

            [array removeObject:str];

        }

     }

    第三种方法是对数组逆序遍历,查找对应元素后删除

NSEnumerator *enumerator = [array reverseObjectEnumerator];

     for (NSString *str in enumerator) {

        if ([str isEqualToString:@"4"]) {

            [array removeObject:str];

        }

     }

     对于逆序遍历,因为我们逆序遍历时,遇到匹配的元素删除后,位置改变的是遍历过得元素,而没有遍历到的元素位置却没有改变,所以遍历能够正常进行.就不会crash。

   解决方法列了这么几种,本人能力有限,如果有更好建议或者有什么不足,请联系我。

原文地址:https://www.cnblogs.com/weicyNo-1/p/7217578.html