iOS原理篇(四):关联对象

  • 关联对象的使用场合
  • 关联对象的基本使用
  • 关联对象的底层原理

一、关联对象的使用场合
默认情况下,因为分类底层结构的限制,不能添加成员变量到分类中,但可以通过关联对象来间接实现。

关联对象提供了以下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)

在添加关联对象的方法中有一个policy属性,它是一个枚举值,对应我们平时定义属性时设置的修饰词:

二、关联对象的基本使用
使用关联对象需要#import

// 给DJTPerson创建一个分类,添加一个name属性
@interface DJTPerson (Test)
@property (nonatomic, copy) NSString *name;
@end

@implementation DJTPerson (Test)
/**
* 第一种写法:创建一个void*类型的指针作为key,它存着自己的地址,只要唯一就行
*/
static const void *DJTNameKey = &DJTNameKey;
- (void)setName:(NSString *)name
{
    objc_setAssociatedObject(self, DJTNameKey, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)name
{
    return objc_getAssociatedObject(self, &DJTNameKey);
}

/**
* 第二种写法:既然是void*类型,使用char类型的地址,节省空间
*/
static const char DJTNameKey;
- (void)setName:(NSString *)name
{
    objc_setAssociatedObject(self, &DJTNameKey, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)name
{
    return objc_getAssociatedObject(self, &DJTNameKey);
}

/**
* 第三种写法:使用属性名作为key,其实也是传的内存地址
*/
- (void)setName:(NSString *)name
{
    objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)name
{
    return objc_getAssociatedObject(self, @"name");
}

/**
* 第四种写法:使用get方法的@selector作为key,这种写法的好处是它返回的是一个结构体指针,写错方法名会有错误提示
*/
- (void)setName:(NSString *)name
{
    objc_setAssociatedObject(self, @selector(name), name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)name
{
    //@selector(name)等价于隐式参数_cmd
    return objc_getAssociatedObject(self, @selector(name));
   // return objc_getAssociatedObject(self, _cmd);
}
@end

三、关联对象的底层原理
关联对象并不是存储在被关联对象本身的内存中,通过分析底层实现,它存储在由AssociationsManager管理的全局统一的一个AssociationsHashMap中,关系如下:


从上图可以看出,关联对象的底层实现依赖下面四个核心对象:

  • AssociationsManager
  • AssociationsHashMap
  • ObjectAssociationMap
  • ObjectionAssociation

它们之间的关系:AssociationsHashMap里存储着某个对象的关联对象Map表,即ObjectAssociationMap,这个表存储了多个关联对象,因为在分类里可以给对象添加多个属性,也就要设置多个关联对象,ObjectAssociationMap中就是我们添加的关联对象,比如name,由ObjectionAssociation存储值和策略,当我们将关联对象(即value值,本例中是person.name = nil)设为nil时,AssociationsMap自动删除这条关联对象;当我们调用objc_removeAssociatedObjects(id object)方法时,就是移除某个对象的所有关联对象,即上图中AssociationHashMap需要移除对象的关联对象Map表;

你可能感兴趣的:(iOS原理篇(四):关联对象)