iOS 开发之避坑指南 (不定期更新)

最近在用 Swift 写自己的新项目 Fontzar, 撸了个遍历时移除不符合条件的值的码然后 build and run, 崩了!没错是崩了!报了个Fatal error: Index out of range

var arr = ["a","c","a"]

for (idx, value) in arr.enumerated() {
    if value == "a" {
        arr.remove(at: idx)
    }
}

print(arr)

同样代码用 OC 写不会蹦!?!

NSMutableArray *arr = @[@"a",@"c",@"a"].mutableCopy;

[arr enumerateObjectsUsingBlock:^(NSString* str, NSUInteger idx, BOOL * _Nonnull stop) {
    if ([str isEqualToString:@"a"]) {
        [arr removeObjectAtIndex:idx];
    }
}];

NSLog(@"%@", tempMArr);

我觉得在 Swift 中崩溃是因为 Swift 是安全的,让我们提前发现了问题。

Swift的特点: 快速 现代 安全 互动

注意: 遍历中不能直接操控原数组,因为这会改变数组长度导然后导致隐藏 Bug,你检查错误都不知哪出问题了(请看下面的代码)。

NSMutableArray *tempMArr = @[@"a",@"x",@"d",@"c",@"a",@"j",@"a",@"c",@"a",@"k",@"f"].mutableCopy;
NSArray *tempArr = @[@"a",@"b",@"c"];

[tempMArr enumerateObjectsUsingBlock:^(NSString* str, NSUInteger idx, BOOL * _Nonnull stop) {
    
    BOOL isExisted = NO;
    
    for (NSString *str2 in tempArr) {
        if ([str isEqualToString:str2]) {
            isExisted = YES;
            break;
        }
    }
    
    if (!isExisted) {
        [tempMArr removeObjectAtIndex:idx];
    }
}];
    
NSLog(@"tempMArr:%@", tempMArr);

控制台打印出的结果:

tempMArr:(
    a,
    d,
    c,
    a,
    a,
    c,
    a,
    f
)

从打印出来的结果我们发现数组中多了 fd,这结果是不符合我们预期。从而证明了遍历中不能操作原数组的真理!

正确姿势:

  • 遍历中把符合条件的值保存在新的数组里
  • 遍历拷贝的数组,操作原数组
  • 倒序遍历数组
  • 使用 Filter 函数filter(_:) Or filteredArrayUsingPredicate:

PS: 如果你用 Swift 开发时遇到了这个问题,可能是因为你用 OC 时这么玩过,或者是你曾经所看过的《iOS 中如何遍历数组?》系列的 Blog 本身就有问题,误导你了或没提醒你。提醒一下 除了 OC 以外的编程语言中就算允许你遍历中操作原数组(不崩溃)也全力避免这种骚操作。最后记得审一下你的 OC 代码

你可能感兴趣的:(iOS 开发之避坑指南 (不定期更新))