说说 Objc Runtime 中的 Associated Objects

swift 发布于 2017年12月18日
Associated Objects

Associated Objects 的一个最常见的应用就是给已有的类添加属性。 如果你使用 Objective-C 语言的话,可以用 Category 给某个已经存在的类添加方法。 但是 Category 却不能给这个类添加实例变量。 比如我们想给 UIView 实现一个字符串类型的 tag(UIView 默认的 tag 实现只能添加 int 类型的值),按照现有的语法就没办法存储这个 tag 的实例变量。

你可以这样实现 UIView 的 Category:

@interface UIView (StringTag)

@property (nonatomic, strong) NSString *stringTag;

@end

这时,我们声明一个 UIView 并给他的 stringTag 属性赋值:

UIView *view = [[UIView alloc] init];
view.stringTag = @"map";

语法和编译都正常,但运行的时候,控制台中就会提示这样的信息:

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[UIView setStringTag:]: unrecognized selector sent to instance 0x7fda52d9a2c0'

程序崩溃了。 虽然我们声明了属性, 但 @property 实际上还是会调用 [UIView setStringTag:] 方法设置实例变量, 但 Category 中是不支持实例变量的,所以就会导致程序在运行时崩溃。 这个例子也能帮助大家加深对 Objc Runtime 的了解。

那么,我们想在 Category 中也能添加实例变量的话该怎么办呢? 这时就可以使用 Associated Object 机制, 在运行时为我们的 UIView 关联属性,将刚刚定义的 Category 实现稍加修改即可达成目标:

@implementation UIView (StringTag)

- (void)setStringTag:(NSString *)stringTag {

objc_setAssociatedObject(self, @selector(stringTag), stringTag, OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

- (NSString *)stringTag {

return objc_getAssociatedObject(self, @selector(stringTag));

}

@end

这样,我们在 Category 中使用 Assocaited Object 在运行时完成了实例变量的设置。 这时候再次运行程序, 刚才对 UIView 相关操作的代码就能够顺利的执行啦。

objc_setAssociatedObject 方法接受 4 个参数, 第一个参数是要将关联值设置到的实例, 第二个是这个关联值的标识 key, 第三个是这个关联值自身, 第四个是这个关联属性的处理方式。

我们注意到它的第二个参数,这里用了 @selector(stringTag) 这种形式。 虽然这个参数代表的是一个 key, 但它的类型不是 NSString, 而是一个 void * 的指针,这就意味着我们可以将任何符合这个类型条件的值设置给它。 @selector(stringTag) 就符合这样的类型,所以我们不必再定义一个变量来表示这个关联值的 key, 只需要将和他属性名对应的 Selector 传递进来即可。

同样的,objc_getAssociatedObject 用于获取关联属性的名称。 它接受两个参数, 其中包括关联值所在的对象,以及它的标识 key。

总结

这里用最简单的篇幅给大家介绍了 Associated Object 在我们平时开发中的应用,对 Category 属性的这个应用很常见,比如著名的框架 AFNetworking 里面就对很多 Category 中的属性定义使用了 Associated Object。 大家有兴趣可以参看它对 UIImage 的 Categoy 定义。 关于 Assoicated Object 更详细的内容,这里的篇幅肯定没有介绍全面,大家还可以参看苹果官方关于 Objective-C Runtime 的文档深入了解。


如果你觉得这篇文章有帮助,还可以关注微信公众号 swift-cafe,会有更多我的原创内容分享给你~

本站文章均为原创内容,如需转载请注明出处,谢谢。
关注微信公众号
发现更多精彩
swift-cafe