最近在用 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
)
从打印出来的结果我们发现数组中多了 f
和 d
,这结果是不符合我们预期。从而证明了遍历中不能操作原数组的真理!
正确姿势:
- 遍历中把符合条件的值保存在新的数组里
- 遍历拷贝的数组,操作原数组
- 倒序遍历数组
- 使用 Filter 函数
filter(_:)
OrfilteredArrayUsingPredicate:
PS: 如果你用 Swift 开发时遇到了这个问题,可能是因为你用 OC 时这么玩过,或者是你曾经所看过的《iOS 中如何遍历数组?》系列的 Blog 本身就有问题,误导你了或没提醒你。提醒一下
除了 OC 以外的编程语言中就算允许你遍历中操作原数组(不崩溃)也全力避免这种骚操作。最后记得审一下你的 OC 代码 😊