对元素类型的要求
在 Objective-C 中,NSArray 只能存储对象类型,而不能直接存储基本类型(例如 int)。但是,可以将基本类型封装在 NSNumber 等对象中,然后将这些对象存储在 NSArray 中。这样,enumerateObjectsUsingBlock: 方法中的 obj 仍然是指向这些对象的指针。
objc
NSArray *array = @[@1, @2, @3];
[array enumerateObjectsUsingBlock:^(NSNumber *obj, NSUInteger idx, BOOL *stop) {
int value = [obj intValue];
NSLog(@"Object: %@, Int Value: %d, Address: %p", obj, value, obj);
}];
obj
是一个指针变量,局部变量
obj 和 array 中对应位置的元素指向相同的内存地址,但它们不是同一个指针变量,而是两个指向同一对象的不同变量。
在 -enumerateObjectsUsingBlock: 方法中,块枚举会遍历数组,并将每个元素的指针赋值给 obj。因此,obj 是一个局部变量,在每次迭代时指向数组中当前遍历的元素。尽管它们指向同一对象,但 obj 是块中的局部变量,而数组中的元素是数组中的存储单元。
objc
NSArray *array = @[@"one", @"two", @"three"];
[array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
NSLog(@"Object: %@, obj Address: %p, obj Pointer Address: %p, array[idx] Address: %p, array[idx] pointer Address: %p", obj, obj, &obj, array[idx], &array[idx]);
}];
循环引用问题
在 Objective-C 中使用 enumerateObjectsUsingBlock
时,同样需要注意循环引用问题。enumerateObjectsUsingBlock
会将块传递给枚举方法,如果块捕获了对自身或外部对象的强引用,这可能导致循环引用,尤其是当这个对象也强引用调用该块的对象时。
示例说明
假设有一个类 MyClass
,其实例方法使用 enumerateObjectsUsingBlock
:
objc
@interface MyClass : NSObject
@property (nonatomic, strong) NSArray *array;
@end
@implementation MyClass
- (void)enumerateArray {
[self.array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
// 这里使用了 self
NSLog(@"%@", self);
}];
}
@end
在上述代码中,块捕获了 self
的强引用。如果 self
对象持有 array
,这就形成了一个循环引用,导致 self
无法被释放。
解决方案
使用弱引用来打破循环引用,可以使用 __weak
或 __block
关键字:
objc
@implementation MyClass
- (void)enumerateArray {
__weak typeof(self) weakSelf = self;
[self.array enumerateObjectsUsingBlock:^(id obj, NSUInteger idx, BOOL *stop) {
// 这里使用了弱引用的 self
__strong typeof(weakSelf) strongSelf = weakSelf;
if (strongSelf) {
NSLog(@"%@", strongSelf);
}
}];
}
@end
在这个修改后的代码中,使用 __weak
将 self
的弱引用传递到块内,然后在块内将其转换为强引用以确保在使用过程中对象的生命周期被正确管理。