iOS weak修饰的释放则自动被置为nil的实现原理 - deepindo/DoNote GitHub Wiki
Runtime维护着一个Weak表,用于存储指向某个对象的所有Weak指针, 这个表就是sideTable
在开源文件objc4-706中, 可以看到NSObject.mm类文件中对于SideTable的定义如下:
struct SideTable {
spinlock_t slock;
RefcountMap refcnts;
weak_table_t weak_table;
SideTable() {
memset(&weak_table, 0, sizeof(weak_table));
}
~SideTable() {
_objc_fatal("Do not delete SideTable.");
}
void lock() { slock.lock(); }
void unlock() { slock.unlock(); }
// Address-ordered lock discipline for a pair of side tables.
template<bool HaveOld, bool HaveNew>
static void lockTwo(SideTable *lock1, SideTable *lock2);
template<bool HaveOld, bool HaveNew>
static void unlockTwo(SideTable *lock1, SideTable *lock2);
};
SideTable 结构体重定了几个非常重要的变量。
// The order of these bits is important.
#define SIDE_TABLE_WEAKLY_REFERENCED (1UL<<0)
#define SIDE_TABLE_DEALLOCATING (1UL<<1) // MSB-ward of weak bit
#define SIDE_TABLE_RC_ONE (1UL<<2) // MSB-ward of deallocating bit
#define SIDE_TABLE_RC_PINNED (1UL<<(WORD_BITS-1))
#define SIDE_TABLE_RC_SHIFT 2
#define SIDE_TABLE_FLAG_MASK (SIDE_TABLE_RC_ONE-1)
Weak表是Hash表,Key是所指对象的地址,Value是Weak指针地址的数组
在对象被回收的时候,经过层层调用,会最终触发下面的方法将所有Weak指针的值设为nil。
runtime源码,objc-weak.m 的 arr_clear_deallocating 函数
weak指针的使用涉及到Hash表的增删改查,有一定的性能开销.