Category关联对象
2019-04-20 本文已影响0人
春风依旧
OC中的分类严格来说,是不能添加对象;但是我们可以通过OC运行是的机制,动态为分类添加属性
一、类中的属性
当在类中添加属性的时候,这个属性做了三件事:
在.h中
@interface ZMJPerson : NSObject
{
int _age;
}
-(void)setAge:(int)age;
-(int)age;
/*
1、生成带下划线的成员变量
2、setter方法的声明与实现
3、getter方法的声明与实现
*/
//@property (nonatomic, assign) int age;
@end
在.m中
-(void)setAge:(int)age{
_age = age;
}
-(int)age{
return _age;
}
二、分类中的属性
category中也可以添加属性,只不过@property只会生成setter和getter的声明,不会生成setter和getter的实现以及成员变量。
给分类中的属性赋值,会崩溃:
unrecognized selector sent to instance 0x102810b10'
如果在分类中实现属性的功能:
1、设置全局变量,把传进去的值存起来
存在问题:创建多个对象,这个属性共用一个全局变量
2、使用可变的字典
存在问题:
①、内存存储问题,一个存在person对象里面,一个存在全局的变量里面;
②、线程安全问题,可以加锁解决
③、添加不同属性,设置不同的字典
3、使用runtime给分类添加属性
三、给分类添加属性
1、使用runtime给分类添加属性,常用的API如下:
1、添加关联对象
void objc_setAssociatedObject(id object, const void * key,
id value, objc_AssociationPolicy policy)
2、获得关联对象
id objc_getAssociatedObject(id object, const void * key)
3、移除所有的关联对象
void objc_removeAssociatedObjects(id object)
4、关联策略
/** 关联对象的参数
@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
*/
2、关联对象的实际运用中的Key
- 2.1 赋值的key : static void *ZMJName_key = &ZMJName_key;
static void * ZMJName_key = & ZMJName_key; //赋值
objc_setAssociatedObject(obj, ZMJName_key, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, ZMJName_key)
- 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: 仅仅是一个指针, static char MyKey; 声明一个字符,&MyKey 字符的地址指针
- 2.3 使用属性名作为key : #define ZMJNameKey @"name";
objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @"property");
NSString *str = @"name"; 其实@"name"就是自身的内存地址:唯一性,可以打印 str,内存地址是一样的
- 2.4 使用get方法的@selecor作为key:
可以使用_cmd来代替@selector(getter)
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))
三、分类的实现原理
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;
}
2、关联对象的原理图
- 关联对象并不是存储在被关联对象本身内存中,所以objc对象销毁后,获取关联的value值就会崩溃。 因为value不是弱引用,不会置为nil,
- 关联对象存储在全局的统一的一个AssociationsManager中
- 设置关联对象为nil,就相当于是移除关联对象
3、移除关联对象
- 移除关联对象:
person2.degree = nil;
- 移除所有的关联对象
void objc_removeAssociatedObjects(id object)
- 当object 被移除的同时对应上面的AssociationsHashMap也会被对应的移除,字典(看到Map可以当做字典来看待)是一对一的关系。
4、移除关联对象