1、前言
在之前学习到分类Category的时候,我们也了解到不能够直接在分类文件中添加成员变量。因为虽然Category里面含有属性项,但却不含成员变量项,不能自动实现对应的get、set方法。
那么要实现在Category中添加成员变量,我们可以通过关联对象去实现。
2、关联对象方法
// 添加关联对象
// object:你要给哪个对象添加关联对象,如果是给当前对象添加关联属性的话,就可以填self
// key:void * 是表示是一个指针,要保持唯一
// value:是要关联的值,只能放对象
// policy:关联策略,对应于value的值的类型
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)
对于其中key值这项呢,需要保持唯一性,一般设置的时候,我们可以采取下面几种策略:
// 方法一:
static void *MyKey = &MyKey;
objc_setAssociatedObject(obj, MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, MyKey)
// 方法二:
static char MyKey;
objc_setAssociatedObject(obj, &MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, &MyKey)
// 方法三:
使用属性名作为key
objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @"property");
// 方法四:
使用get方法的@selecor作为key
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))
对于其中的policy值这项呢,也有其对应的规则:
OBJC_ASSOCIATION_ASSIGN assign
OBJC_ASSOCIATION_RETAIN_NONATOMIC strong, nonatomic
OBJC_ASSOCIATION_COPY_NONATOMIC copy, nonatomic
OBJC_ASSOCIATION_RETAIN strong, atomic
OBJC_ASSOCIATION_COPY copy, atomic
3、关联对象的使用
有类Person,其已有一个属性age,在他的分类中,我们定义了属性name和weight。
下面我们要用关联对象去实现对name和weight的使用。
#pragma mark -MJPerson+Test.h文件
#import "MJPerson.h"
@interface MJPerson (Test)
@property(nonatomic,assign)int weight;
@property (copy, nonatomic) NSString *name;
@end
#pragma mark -MJPerson+Test.m文件
#import "MJPerson+Test.h"
#import <objc/runtime.h>
@implementation MJPerson (Test)
-(void)setName:(NSString *)name{
objc_setAssociatedObject(self, @selector(name), name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
- (NSString *)name{
return objc_getAssociatedObject(self, @selector(name));
}
-(void)setWeight:(int)weight{
objc_setAssociatedObject(self, @selector(weight), @(weight), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(int)weight{
return [objc_getAssociatedObject(self, _cmd) intValue];
}
@end
这样通过关联对象就能实现对分类文件中属性的使用了。
MJPerson *person = [[MJPerson alloc] init];
person.age = 10;
person.name = @"haha";
person.weight = 120;
MJPerson *perso2 = [[MJPerson alloc] init];
perso2.age = 20;
perso2.name = @"hdddaha";
perso2.weight = 10;
NSLog(@"name = %@,age = %d,weight = %d",person.name,person.age,person.weight);
NSLog(@"name = %@,age = %d,weight = %d",perso2.name,perso2.age,perso2.weight);
//打印结果:
==================================================
name = haha,age = 10,weight = 120
name = hdddaha,age = 20,weight = 10
4、关联对象的实现原理
如下图: 关联对象的原理AssociationsManager里面是由一个静态AssociationsHashMap来存储所有的关联对象的,这相当于把所有对象的关联对象都存在一个全局HashMap里面。
HashMap里的key值是这个对象的指针地址(任意两个不同对象的指针地址一定是不同的),value值是ObjectAssociationsMap,里面保存了关联对象的kv对。
网友评论