iOS底层原理总结 - 探寻关联对象本质

本篇主要是对小码哥底层视频学习的总结。方便日后复习。
上一篇《iOS底层原理总结 - 探寻Category的本质》:
//www.greatytc.com/p/16bf93ffcd6c

本篇学习总结:

  • 添加关联对象
  • 探寻关联对象本质

好了,带着问题,我们一一开始阅读吧 😊

一.添加关联对象

1.面试题:分类可以添加成员变量吗?

我们在上一篇文章iOS底层原理总结 - 探寻Category本质中已经讲过了,分类代码转化成c++底层代码后会生成一个category_t 结构体变量

_category_t结构体信息.png

category_t 结构体中没有存储成员变量的信息,所以严格上来说,分类不可以添加成员变量,只能添加属性,分类添加属性,系统会自动生成setter跟getter方法声明,不会生成属性对应的成员变量以及setter跟getter方法的实现。

我们知道类的成员变量,属性信息是存储在类对象中的class_rw_t 结构体中,成员变量的数据信息是存储在实例对象中,分类中添加属性,不会生成成员变量,所以即使手动实现了setter跟getter方法,存储的数据也不会存储到实例对象的成员变量信息中,我们只能是存储到一个全局变量中,类似如下代码:

static NSString *_name;
-(void)setName:(NSString *)name
{
    _name = name;
}
-(NSString *)name
{
    return _name;
}
NSObject *objc1 = [[NSObject alloc]init];
objc1.name = @"objc1";
NSObject *objc2 = [[NSObject alloc]init];
objc2.name = @"objc2";
NSLog(@"objc1.name = %@,objc2.name = %@",objc1.name,objc2.name);
//打印结果:
objc1.name = objc2,objc2.name = objc2

这样导致的后果是,我们创建多个实例对象,每个实例对象存储的name数据不会一一对应,因为static NSString *_name 放到类对象的数据中,类对象有且只有一个,存储name数据以最后一个对象数据存储为准

如果我们想分类中的属性可以通过点语法使用,怎么做到呢?我们通过添加关联对象来处理。

我们利用runtime给分类属性添加关联对象,runtime中提供了动态添加属性和获取属性的方法

-(void)setName:(NSString *)name
{
    objc_setAssociatedObject(self, @"name",name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(NSString *)name
{
    return objc_getAssociatedObject(self, @"name");    
}

下面我们分析一下动态添加关联对象的api

  1. objc_setAssociatedObject
objc_setAssociatedObject(id object, const void *key,
 id value, objc_AssociationPolicy policy);

参数一:id object:给哪个对象添加关联对象数据,就是哪个对象,参数可以是实例方法,也可以是类方法,这里我们用self(代表调用方法者,也就是当前的实例对象)。
参数二:const void key:添加关联对象存取数据的key,为了保证每个对象每个属性的存储数据唯一性,这里我们一般采用一个指针@selector(name)=_cmd,及方法的内存地址作为存储数据key。
参数三:id value:关联的值,也就是说每个对象每个属性存储的数据
参数四:objc_AssociationPolicy policy:策略,属性以什么形式保存,它是一个枚举:

typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy) {
    OBJC_ASSOCIATION_ASSIGN = 0,  // 指定一个弱引用相关联的对象
    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, // 指定相关对象的强引用,非原子性
    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,  // 指定相关的对象被复制,非原子性
    OBJC_ASSOCIATION_RETAIN = 01401,  // 指定相关对象的强引用,原子性
    OBJC_ASSOCIATION_COPY = 01403     // 指定相关的对象被复制,原子性   
};
属性说明.png
  1. objc_getAssociatedObject
objc_getAssociatedObject(id object, const void *key);

参数一:id object:添加关联数据的对象,就是上述方法的object
参数二:const void key:关联对象中存储数据的key,这里我们通过key获取到之前存储的数据
3.removeAssociatedObjects

- (void)removeAssociatedObjects
{
    // 移除所有关联对象
    objc_removeAssociatedObjects(self);
}

移除当前对象关联的所有的数据

通过关联对象方法间接实现了NSObject添加name属性后可以通过点语法为属性赋值,取值。

NSObject *objc1 = [[NSObject alloc]init];
objc1.name = @"objc1";
NSObject *objc2 = [[NSObject alloc]init];
objc2.name = @"objc2";
NSLog(@"objc1.name = %@,objc2.name = %@",objc1.name,objc2.name);
//打印结果如下:
objc1.name = objc1,objc2.name = objc2
//可以看出 每一个objc对象存储着各自赋值的name数据

二.探索关联对象本质

实现关联对象技术的核心对象有:

  • 1.AssociationsManager
  • 2.AssociationsHashMap
  • 3.ObjectAssociationMap
  • 4.ObjcAssociation

其中xxxMap 就可以理解为我们OC中的字典对象,通过key-value进行赋值
我们通过源码来探寻关联对象技术的核心对象存在形式以及其作用。

1.objc_setAssociatedObject函数

打开下载好的objc源代码,搜索找到objc_setAssociatedObject函数,看一下函数实现

objc_setAssociatedObject函数实现.png

里面调用了_object_set_associative_reference函数,实现如下:

_object_set_associative_reference函数内部实现.png

_object_set_associative_reference函数内部我们可以在全部找到我们上面说过的实现关联对象的核心对象,接下来我们来一个一个看其内部实现原理探寻他们之间的关系。

  • AssociationsManager

通过AssociationsManager内部源码发现,AssociationsManager内部有一个AssociationsHashMap对象

AssociationsHashMap内部.png

通过AssociationsHashMap内部源码我们可以发现AssociationsHashMap继承自unordered_map,首先看一下unordered_map内部源码

unordered_map内部分源码.png

从unordered_map源码中我们可以看出_key和_Tp也就是前两个参数对应着map中的key和value,那么对照上面的AssociationsHashMap源码发现_key中传入的是disguised_ptr_t,_Tp中传入的值则为ObjectAssociationMap* 。
紧接着我们来到ObjectAssociationMap中,上图中ObjectAssociationMap已经标记出,我们发现ObjectAssociationMap中同样以key,value的方式存储着ObjcAssociation,接着我们来到ObjcAssociation中,可以看到如下结构:

ObjcAssociation内部实现.png

我们发现ObjcAssociation存储着_policy,_value,而这两个值我们可以发现是我们调用objc_setAssociatedObject函数传入的值,也就是说我们在调用objc_setAssociatedObject函数中传入的value和policy这两个值最终存储在ObjcAssociation中。

现在我们已经对AssociationsManagerAssociationsHashMapObjectAssociationMapObjcAssociation 四个对象之间的关系有了简单的认识,那么接下来我们细读源码,看一下objc_setAssociatedObject函数中传入的四个参数分别放到哪个对象中充当什么作用。

让我们重新在看一下_object_set_associative_reference函数实现

_object_set_associative_reference函数内部实现.png

细读上述源码 我们可以发现,首先根据我们传入的value经过acquireValue函数处理获取new_value,acquireValue函数内部其实是通过对策略的判断返回不同的值

acquireValue函数内部实现.png

之后创建AssociationsManager manager,以及拿到manager内部的AssociationsHashMap及associations。
之后我们看我们传入的第一个参数object,object经过DISGUISE函数被转化成为了disguised_ptr_t类型的disguised_object。

DISGUISE函数.png

DISGUISE函数其实仅对object内存地址做了位运算,并不强制引用object对象。

之后我们看到被处理成new_value的value,同policy被存入ObjcAssociation中,而ObjcAssociation对应我们传入的key被存入了ObjectAssociationMap中,disguised_object和ObjectAssociationMap则以key-value的形式对应存储到associations中,也就是AssociationsHashMap中。

关键代码.png

如果我们设置value为nil的话,就会执行下面的代码

value=nil 移除关联对象.png

从上述代码中我们可以看出,如果我们主动给value设置为nil,就会将关联对象从ObjectAssociationMap中移除。

最后我们通过一张图可以很清晰的理清楚其中的关系

关联对象底层关系图.png

通过上图我们可以总结为:一个实例对象就对应一个ObjectAssociationMap,而ObjectAssociationMap中存储着多个此实例对象关联对象的key以及ObjcAssociation,而ObjcAssociation中存储着关联对象的value和policy策略。
由此我们可以知道关联对象并不是放在了原来的对象里面,而是自己维护了一个全局的hashmap,用于存放每一个对象及其对应关联属性表格。

2.** objc_getAssociatedObject函数**

objc_getAssociatedObject 内部调用的是_object_get_associative_reference

objc_getAssociatedObject内部函数实现.png

我们看一下_object_get_associative_reference函数具体实现

_object_get_associative_reference.png

从_object_get_associative_reference函数内部可以看出,向set方法中那样,反向将value一层一层取出最后return出去。

3.objc_removeAssociatedObjects函数

objc_removeAssociatedObjects函数用于删除所有的关联对象,objc_removeAssociatedObjects函数内部调用的是_object_remove_assocations函数

objc_removeAssociatedObjects.png

我们来看_object_remove_assocations函数内部实现

_object_remove_assocations函数内部实现.png

上述源码我们可以看出_object_remove_assocations函数将object对象对应的所有关联对象全程删除。

此时我们在回来看objc_AssociationPolicy policy参数,这是一个枚举类型的数据

typedef OBJC_ENUM(uintptr_t, objc_AssociationPolicy) {
    OBJC_ASSOCIATION_ASSIGN = 0,  // 指定一个弱引用相关联的对象
    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, // 指定相关对象的强引用,非原子性
    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,  // 指定相关的对象被复制,非原子性
    OBJC_ASSOCIATION_RETAIN = 01401,  // 指定相关对象的强引用,原子性
    OBJC_ASSOCIATION_COPY = 01403     // 指定相关的对象被复制,原子性   
};

我们会发现其中只有RETAIN和COPY二而没有weak呢,通过上面对源码的分析我们知道,object经过DISGUISE函数被转化成了disguised_ptr_t类型的disguised_object

disguised_ptr_t disguised_object = DISGUISE(object);

而同时我们知道,weak修饰的属性,当没有拥有对象之后就会被销毁,并且指针置位nil,那么在对象销毁之后,虽然在map中既然存在值object对应的AssociationsHashMap,但是因为object地址已经被置位nil,会造成坏地址访问而无法根据object对象的地址转化为disguised_object了。

总结本篇面试题:

  • 1.面试题:分类可以添加成员变量吗?

分类不可以添加成员变量,可以添加属性,但是系统只会自动生成setter跟getter方法声明,不会生成属性对应的成员变量,也不会生成setter跟getter方法的实现,如果我们用点语法使用分类中的属性,通过在分类中手动实现setter跟getter方法中关联属性对象。

本篇学习先记录到此,感谢阅读,如有错误,不吝赐教。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 207,113评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,644评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,340评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,449评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,445评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,166评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,442评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,105评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,601评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,066评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,161评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,792评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,351评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,352评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,584评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,618评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,916评论 2 344