0

虽然2020年我还在问关于ObjC的问题有点愚蠢,但请耐心和体谅...

我正在阅读 BloksKit 的源代码并遇到了一个奇怪的情况。

#import <objc/runtime.h>

@interface _WeakAssociatedObjectWrapper : NSObject
@property (nonatomic, weak) id object;
@end

@implementation _WeakAssociatedObjectWrapper
@end

@interface NSObject (AddWeak)
@end

@implementation NSObject (AddWeak)
- (void)setWeakProp:(id)weakProp {
    _WeakAssociatedObjectWrapper *wrapper  = objc_getAssociatedObject(self, @selector(weakProp));
    if (!wrapper) {
        wrapper = [[_WeakAssociatedObjectWrapper alloc] init];
        objc_setAssociatedObject(self, @selector(weakProp), wrapper, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    wrapper.object = weakProp;
}
- (id)weakProp {
    id value = objc_getAssociatedObject(self, _cmd);
    if ([value isKindOfClass:_WeakAssociatedObjectWrapper.class]) {
        return [(_WeakAssociatedObjectWrapper *)value object];
    }
    return value;
}
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        NSObject *obj = [[NSObject alloc] init];
        {
            NSObject *prop = [[NSObject alloc] init];
            [obj setWeakProp:prop];

            [obj weakProp]; // *Weird!!
        }
        NSLog(@"Now obj.weakProp = %@", [obj weakProp]);
    }
    return 0;
}

此代码正在为类别添加一个弱关联对象。(BlocksKit 这样做)

注意*Weird!!线。如果此行被注释掉,那么它会打印(null),这是合理的,因为prop在范围之外被释放{}。另一方面,如果没有注释掉,它会打印<NSObject: 0xxxxx>,这表明它prop以某种方式被某人保留(或任何其他原因?)。这里发生了什么??!(BlocksKit 的行为相同!)

环境:XCode 10.3

4

1 回答 1

0

这是一个特点。对于这种情况(以及任何类似情况)

[obj 弱属性];

通过属性/访问器命名约定 ARC 返回自动释放的实例,因此在您的情况下@autoreleasepool保留它,如下所示的测试可以显示这一点。

int main(int argc, const char * argv[]) {
    NSObject *obj = [[NSObject alloc] init];
    @autoreleasepool {
        {
            NSObject *prop = [[NSObject alloc] init];
            [obj setWeakProp:prop];

            [obj weakProp]; // *Weird!!
        }
        NSLog(@"Now obj.weakProp = %@", [obj weakProp]);
    }
    NSLog(@"After autoreleased >> obj.weakProp = %@", [obj weakProp]);
    return 0;
}
于 2020-01-16T07:56:28.477 回答