我们想对一个已经存在的类增加属性的时候,有时候我们会选择写分类,但是既然类都给我们写好了,如果每次仅因为需要增加一个属性的时候而却写一个分类文件,未免就显得太麻烦了。
这时候我们可以动用一点runtime的知识,去动态关联对象的属性。
主要方法:
objc_setAssociatedObject(<id object>, <const void *key>, <id value>, <objc_AssociationPolicy policy>)
//设定关联对象,key可以当做一个标识。
//(下文称为ObjCSet方法)
objc_getAssociatedObject(id object, const void *key)
//(下文称为ObjCGet方法)
//通过key获取关联的对象
objc_removeAssociatedObjects(id object)
//解除所有关联对象,若想解除指定对象,可以重写一遍setAssociate方法,关键字不变,关联对象写为nil即可。
与KVC 方法类似,两个方法都是通过特定的Key与对象关联,policy就相当于属性修饰符。
Object Association Types
|Object Association Types | 等价的属性修饰符 |
| ------------- |:-------------:| -----:|
| OBJC_ASSOCIATION_ASSIGN | assign |
| OBJC_ASSOCIATION_RETAIN_NONATOMIC | nonatomic, retain |
| OBJC_ASSOCIATION_COPY_NONATOMIC | nonatomic, copy |
| OBJC_ASSOCIATION_RETAIN | retain |
| OBJC_ASSOCIATION_COPY | copy |
上面这张表格中,左边的修饰关键字与右边是等价的。
因此我们在做动态的对象关联时候,要注意使用这些关键字,避免内存引用循环,使用上来说,大体与使用属性修饰符是差不多的。
与KVC方法最大的不同之处在于,这个ObjCSet和Get方法使用的是指针作为自己的对象的key而不是通常意义上的字符串。
所以采用此方法时,我们的key应该是一个全局的常量。
当然多重关联也是可以的:
多重关联也是可以的:
/**与Button进行关联*/
objc_setAssociatedObject(sender, ASS_BUTTON_KEY, self.lblTextOne.text, OBJC_ASSOCIATION_ASSIGN);
/**与Navi进行关联*/
objc_setAssociatedObject(self.navigationController, ASS_NAVI_KEY, sender, OBJC_ASSOCIATION_ASSIGN);
objc_setAssociatedObject(self.navigationController, ASS_MT_KEY, self.lblTextTwo.text, OBJC_ASSOCIATION_ASSIGN);
就是单纯的三个方法,还有一些关于修饰关键字的设定问题,其余并没有大问题,不多做demo。
总结
1 利用runtime中的objc_setAssocaite和objc_getAssocaite方法可以将两个对象进行动态关联。
2 objc_set时那些内存管理关键词的意思与属性设置时的修饰词类似,表中已列出等价关系
3 这种方法的使用会使维护变的困难,当有更好方法时建议不要采用。