在微博上看到两道ios的面试题,请指出一下两段代码中的问题:
第一段代码:
@property (copy) NSMutableArray<NSImage *>*photos;
- (void)replacewithStockPhoto:(NSImage*)stockPhoto {
self.photos = [NSMutableArray<NSImage *> new];
[self.photos addObject:stockPhoto];
}
第二段代码:
- (BOOL)validateDictionary:(NSDictionary*)dict
usingChecker:(checker*)checker
error:(NSError **)error {
__block BOOL isValid = YES;
[dict enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull obj, BOOL * _Nonnull stop) {
if ([checker checkObject:obj forKey:key]) return;
*stop = YES; isValid = NO;
if (error) *error = [NSError errorWithDomain...];
}];
return isValid;
}
说说我的理解,第一道题,copy
是会复制一个不可变的数组,所以photos
真正的类型是NSArray
而不是NsMutableArray
,因此在调用addObject:
方法会崩溃.
第二道题参数NSError**
会被编译器重写为 NSError * __autoreleasing * (ARC特性)
,在dict内部遍历的block中,会添加一个内部的autoreleasepool
,所以,在内部遍历部分的原代码相当于:
....
for(NSUInteger index = 0,index < dict.allKeys.count && !stop;index++) {
@autoreleasepool{
...
if(error) *error = [NSError errorWithDomain...];
}
}
...
被autoreleasepool
修饰的NSError
对象在离开autoreleasepool
时会被释放,这样外部访问NSError
对象时,就会出现EXC_BAD_ACCESS
错误.解决方法,可以在遍历前,新建一个临时的NSError
,在遍历后,对传入的error
赋值.