今天看了FDTemplateLayoutCell的源码,类别里面相当频繁使用了关联对象,做笔记!!!学套路
主要函数:
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy); id objc_getAssociatedObject(id object, const void *key); void objc_removeAssociatedObjects(id object);
基本说明:关联对象就是runTime界的NSMultableDictionary
objc_setAssociatedObject 相当于 setValue:forKey 进行关联value对象 objc_getAssociatedObject 用来读取对象 objc_AssociationPolicy 属性 是设定该value在object内的属性,即 assgin, (retain,nonatomic)...等 objc_removeAssociatedObjects 函数来移除一个关联对象,或者使用objc_setAssociatedObject函数将key指定的关联对象设置为nil。
相关参数
key:要保证全局唯一,key与关联的对象是一一对应关系。必须全局唯一。通常用@selector(methodName)作为key。
value:要关联的对象。
policy:关联策略。有五种关联策略。
OBJC_ASSOCIATION_ASSIGN 等价于 @property(assign)。
OBJC_ASSOCIATION_RETAIN_NONATOMIC等价于 @property(strong, nonatomic)。
OBJC_ASSOCIATION_COPY_NONATOMIC等价于@property(copy, nonatomic)。
OBJC_ASSOCIATION_RETAIN等价于@property(strong,atomic)。
OBJC_ASSOCIATION_COPY等价于@property(copy, atomic)。
使用场景:关联对象相当于实例变量,在类别(也有人管叫分类)里面,不能创建实例变量, 关联对象就可以解决这种问题。(对应属性,有对应属性的的runtime 解决方法,本篇文章主讲 关联对象学习)
基本使用:举例创建 UIButton类别
// // UIButton+Block.h // SectionDemo // // Created by HF on 2017/7/13. // Copyright © 2017年 HF-Liqun. All rights reserved. // #import <UIKit/UIKit.h> typedef void (^btnBlock)(id sender); @interface UIButton (Block) - (void)handelWithBlock:(btnBlock)block; @end // // UIButton+Block.m // SectionDemo // // Created by HF on 2017/7/13. // Copyright © 2017年 HF-Liqun. All rights reserved. // #import "UIButton+Block.h" #import <objc/runtime.h> @implementation UIButton (Block) - (void)handelWithBlock:(btnBlock)block { if (block) { objc_setAssociatedObject(self,@selector(btnAction:) , block, OBJC_ASSOCIATION_RETAIN_NONATOMIC); } [self addTarget:self action:@selector(btnAction:) forControlEvents:UIControlEventTouchUpInside]; } - (void)btnAction:(id)sender { btnBlock block = objc_getAssociatedObject(self,@selector(btnAction:)); if (block) { block(sender); } } @end
注意到标记亮粉色部分 是第二个参数 const void *key 类型通常都是通常都是会采用静态变量来作为关键字 可以自己创建 也可以使用"@selector(btnAction:) "。
自己设置静态变量做关键字:
在FDTemplateLayoutCell 中 第二个参数进一步优化,出现了_cmd 代替了 &btnKey 或者 @selector(methodName).
因为:
_cmd在Objective-C的方法中表示当前方法的selector,正如同self表示当前方法调用的对象实例。
这里强调当前,_cmd的作用域只在当前方法里,直指当前方法名@selector。例如:
这样处理好处是,该方法相对化简了繁文缛节,并且可读性良好。不容易写错key。
参考 FDTemplateLayoutCell中使用_cmd 的使用
至此,设置关联对象关键key,一共有三种写法:
(1)静态变量&btnKey
(2)@selector(methodName)
(3)_cmd