上一篇博客里面写到如何实现给分类添加成员变量?,在这里咱们通过学习关联对象来解决下。
一、关联对象的常用API
默认情况下,因为分类底层结构的限制,不能添加成员变量到分类中。但可以通过关联对象来间接实现
-
添加关联对象
void objc_setAssociatedObject(id object, const void * key, id value, objc_AssociationPolicy policy)-
上面方法的参数解释
/** 关联对象的参数 @param object 设置对象: 在这里是 self(类自己) @param key#> <#key#> description#> @param value#> 关联值:这里是 name @param policy#> 关联策略 @return */ /** 关联策略(objc_AssociationPolicy) 对应的修饰符 OBJC_ASSOCIATION_ASSIGN = 0, assign OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1 strong, nonatomic OBJC_ASSOCIATION_COPY_NONATOMIC = 3, copy, nonatomic OBJC_ASSOCIATION_RETAIN = 01401, strong, atomic OBJC_ASSOCIATION_COPY = 01403 copy, atomic */
-
-
获得关联对象
id objc_getAssociatedObject(id object, const void * key) -
移除所有的关联对象
void objc_removeAssociatedObjects(id object)
二、关联对象的实际运用中的Key
static : 代表只能在此文件内访问
说明:添加几个成员变量就要写几个关联对象的Key
下面四种关联对象的实际使用的demo
-
2.1、赋值的key :
static void *MyKey = &MyKey;static void *MyKey = &MyKey; // 关联对象 objc_setAssociatedObject(obj, MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC) // 获得关联对象 objc_getAssociatedObject(obj, MyKey)demo查看JKPerson+Test -
2.2、内存地址的key :
static char MyKey;static char MyKey; objc_setAssociatedObject(obj, &MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC) objc_getAssociatedObject(obj, &MyKey)const void * _Nonnull key: 仅仅是一个指针,
&MyKey地址就可以,static char MyKey;不是一定要用char,这里只是它的字节是一个字节,在此仅仅只传一个指针而已
demo查看JKPerson+Test2 -
2.3、使用属性名作为key :
#define JKRevelKey @"revel";NSArray *str = @"revel";: 其实@"revel"就是自身的内存地址:唯一性,可以打印str,内存地址是一样的objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC); objc_getAssociatedObject(obj, @"property");demo查看JKPerson+Test3 -
2.4、使用get方法的
@selecor作为key可以使用
_cmd来代替@selector(getter)objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC) objc_getAssociatedObject(obj, @selector(getter))demo查看JKPerson+Test4
三、关联对象的原理探索
-
3.1、实现关联对象技术的核心对象有
AssociationsManagerClass AssociationsManager { static AssociationsHashMap *_map; };AssociationsHashMapclass AssociationsHashMap : public unordered_map<disguised_ptr_t, ObjectAssociationMap *, DisguisedPointerHash, DisguisedPointerEqual, AssociationsHashMapAllocator>ObjectAssociationMapclass ObjectAssociationMap : public std::map<void *, ObjcAssociation, ObjectPointerLess, ObjectAssociationMapAllocator>ObjcAssociationclass ObjcAssociation { uintptr_t _policy; id _value; } -
3.2、下面图表述了上面四个核心对象的关系
关联对象技术的核心对象的关系 -
3.3、移除关联对象和移除所有的关联对象
-
移除关联对象
person2.degree = nil; -
移除所有的关联对象
void objc_removeAssociatedObjects(id object) 可以查看demo里面的ViewController中的test5方法:
当object 被移除的同时对应上面的AssociationsHashMap也会被对应的移除,字典(看到Map可以当做字典来看待)是一对一的关系。
-
