上一篇博客里面写到如何实现给分类添加成员变量?,在这里咱们通过学习关联对象来解决下。
一、关联对象的常用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
- 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、实现关联对象技术的核心对象有
AssociationsManager
Class AssociationsManager { static AssociationsHashMap *_map; };
AssociationsHashMap
class AssociationsHashMap : public unordered_map<disguised_ptr_t, ObjectAssociationMap *, DisguisedPointerHash, DisguisedPointerEqual, AssociationsHashMapAllocator>
ObjectAssociationMap
class ObjectAssociationMap : public std::map<void *, ObjcAssociation, ObjectPointerLess, ObjectAssociationMapAllocator>
ObjcAssociation
class ObjcAssociation { uintptr_t _policy; id _value; }
- 3.2、下面图表述了上面四个核心对象的关系
3.3、移除关联对象和移除所有的关联对象
- 移除关联对象
person2.degree = nil;
- 移除所有的关联对象
void objc_removeAssociatedObjects(id object)
- 可以查看demo里面的ViewController中的test5方法:
当object 被移除的同时对应上面的AssociationsHashMap也会被对应的移除
,字典(看到Map可以当做字典来看待)是一对一的关系。