我们都知道Runtime维护了一个Weak表,用于存储指向某个对象的所有Weak指针。Weak表其实是一个哈希表,Key是所指对象的地址,Value是Weak指针的地址(这个地址的值是所指对象的地址)的数组。
在对象被回收的时候,经过层层调用,会最终触发下面的方法将所有Weak指针的值设为nil
如果在细致去分析的可以大概分为三个步骤:
1,初始化:初始化weak会调用objc_initWeak函数,生成一个新的weak指针指向对象地址。
2,添加引用时:objc_initWeak会去调用objc_storeWeak()函数,它的作用是更新weak指针指向的对象,创建对应的弱引用表
3,释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录
粗略的从代码层面的来分析:
{
id __weak obj_weak = obj;//obj已被赋值,并且是strong类型的
}
/*编译器的模拟代码*/
id obj_weak;
objc_initWeak(&obj_weak,obj);//初始化附有__weak修饰符的变量
objc_destroyWeak(&obj_weak);//释放该变量
其中objc_initWeak objc_destroyWeak都是调用了objc_storeWeak函数,所以,上面的代码可以转化为下面的代码
id obj_weak;
obj_weak = 0;
objc_storeWeak(&obj_weak,obj);
objc_storeWeak(&obj,0);
objc_storeWeak函数以把obj的地址作为键值,obj_weak的地址作为值存放到weak表(weak是一个hash表)中。
释放对象时,废弃对象的同时,程序的动作是怎样的呢?对象通过objc_release释放。
1. objc_release
2. 因为引用计数为0所以执行dealloc
3. _objc_rootDealloc
4. object_dispose
5. objc_destructInstance
6. objc_clear_deallocating
而,调用objc_clear_deallocating的动作如下:
1. 从weak表中获取废弃对象的地址为键值的记录。
2. 将包含在记录中的所有附有__weak修饰符变量的地址,赋值为nil
3. 从weak表中删除记录
4. 从引用计数表中删除废弃对象的地址作为键值的记录
根据以上步骤,前面说的如果附有__weak修饰符的变量所引用的对象被废弃,则将nil赋值给这个变量,这个功能即被实现。
---------------------












网友评论