OC的内存管理方式是使用对象的引用计数为维护, 简单来说就是谁创建, 谁释放, 谁引用谁管理的机制, , 其中关键函数如下:
retain - 引用计数 + 1
release - 引用计数 - 1
retainCount - 引用计数
alloc - 分配内存, 构造isa
dealloc - 释放空间
- 以
alloc、new、copy、mutableCopy创建的对象, 引用计数+1 - 调用对象的
retain方法之后它的引用计数+1 - 调用对象的
release方法之后它的引用计数-1 - 如果一个对象的引用计数为0, 系统会自动调用
dealloc方法释放该对象
在ARC中, 编译器会在编译时, 帮我们插入retain, release, autorelase 等方法, 不用我们手动调用. 但是MRC中需要开发者手动管理.
MRC中, dealloc 方法需要在最后手动调用super.dealloc方法!
- 父类可能被其他对象引用
- 本类最终的释放时在父类中完成
OC对象引用计数存储的位置与弱引用表
OC中对象的引用计数在arm64以后, 存储在对象的isa的最后几个关键的bit:
uintptr_t has_sidetable_rc : 1;
uintptr_t extra_rc : 19
其中extra_rc用来存储该对象的引用计数信息, 如果无法存储下, 需要借助全局的sideTables来存储一半的引用计数值.关于使用extra_rc存储引用计数信息略过, 下面直接看SideTable相关内容
在isa中有一个has_sidetable_rc标记该对象是否拥有SideTable来帮助管理引用计数.
而在iOS系统中, 操作系统会在APP启动时候, 分配一个全局的静态的SideTables的Map!!!! 它是一个HashTable.
在iOS系统中内部存储着8张SideTable表!!!使用一个与OC对象地址相关的Hash函数选择对应的SideTable:
struct SideTable {
spinlock_t slock;//自旋锁
RefcountMap refcnts;//引用计数表 -- RefcountMap其实是个C++的Map, 也是用OC对象地址相关的Hash函数计算的key
weak_table_t weak_table;//弱引用表 -- weak_table_t weak_table
}
struct weak_table_t {
weak_entry_t *weak_entries; // 是一个 weak_entry_t 的 HashTable
size_t num_entries; // 数组中的元素个数
...
};
struct weak_entry_t {
DisguisedPtr<objc_object> referent; //被弱引用的对象, 当这个对象被释放的时候,referrers里的所有指针都会被设置成nil。
// 引用该对象的对象列表
// 联合体: 引用个数小于4用固定数组, >4, 用动态数组
union{
weak_referrer_t *referrers; // 弱引用该对象的对象指针地址的hash数组
weak_referrer_t inline_referrers[WEAK_INLINE_COUNT];
...
}
}
结论:
- SideTables是一个Map, 维护多个SideTable
- OC对象关联的
SideTable维护着对象的引用计数表和一个弱引用表 - 引用计数表实际也是一个Map, 需要用对象的地址再查一次, 才能找到真正的引用计数相关的信息.
- weak_table是全局的弱引用表, referent 是引用对象的指针,
referrer是weak指针的地址, 因此当对象释放时, 会将全部的referrer被设置成nil!!
Apple 设计 SideTables 这样一个HashMap来持有多个SideTable, 为了减少加锁带来的性能损失, 可以简单理解成一个数组SideTables[8]
OC的dealloc方法
在有了以上知识以后, 再看oc的 dealloc方法!!!
前面说到当OC对象被调用release方法时, 发现引用计数为0, 会调用dealloc方法, 该方法的源码如下:
- (void)dealloc {
_objc_rootDealloc(self);
}
void _objc_rootDealloc(id obj){
assert(obj);
obj->rootDealloc();
}
inline void objc_object::rootDealloc(){
if (isTaggedPointer()) return; // fixme necessary?
/*
通过isa判断, 没有以下情况, 直接free!!!
1. 是否拥有弱引用指针指向本对象
2. 是否有关联对象
3. 是否有cxx析构函数
4. 是否有sidetable
*/
if (fastpath(isa.nonpointer &&
!isa.weakly_referenced &&
!isa.has_assoc &&
!isa.has_cxx_dtor &&
!isa.has_sidetable_rc)) {
assert(!sidetable_present());
free(this);
} else {
object_dispose((id)this);
}
}
id object_dispose(id obj) {
if (!obj) return nil;
objc_destructInstance(obj);
free(obj);
return nil;
}
void *objc_destructInstance(id obj) {
if (obj) {
// Read all of the flags at once for performance.
bool cxx = obj->hasCxxDtor();
bool assoc = obj->hasAssociatedObjects();
// This order is important.
if (cxx) object_cxxDestruct(obj);
if (assoc) _object_remove_assocations(obj);
obj->clearDeallocating();
}
return obj;
}
void objc_object::clearDeallocating() {
if (slowpath(!isa.nonpointer)) {
// Slow path for raw pointer isa.
sidetable_clearDeallocating();
} else if (slowpath(isa.weakly_referenced || isa.has_sidetable_rc)) {
// Slow path for non-pointer isa with weak refs and/or side table data.
clearDeallocating_slow();
}
assert(!sidetable_present());
}
void objc_object::clearDeallocating_slow() {
assert(isa.nonpointer && (isa.weakly_referenced || isa.has_sidetable_rc));
/*
处理SideTable, 获取对象关联的SideTable
1. 清理 weak_table 表, 主要是设置成nil, 然后清理weak_table表
2. 引用计数表处理
*/
SideTable& table = SideTables()[this];
table.lock();
if (isa.weakly_referenced) {
weak_clear_no_lock(&table.weak_table, (id)this);
}
if (isa.has_sidetable_rc) {
table.refcnts.erase(this);
}
table.unlock();
}
void weak_clear_no_lock(weak_table_t *weak_table, id referent_id) {
// 调用 dealloc 的对象
objc_object *referent = (objc_object *)referent_id;
// 获取hashTable最终的 weak_entry_t * 指针!
weak_entry_t *entry = weak_entry_for_referent(weak_table, referent);
if (entry == nil) {
/// XXX shouldn t happen, but does with mismatched CF/objc
//printf("XXX no entry for clear deallocating %p\n", referent);
return;
}
// zero out references
weak_referrer_t *referrers;
size_t count;
if (entry->out_of_line()) {
referrers = entry->referrers;
count = TABLE_SIZE(entry);
} else {
referrers = entry->inline_referrers;
count = WEAK_INLINE_COUNT;
}
// 遍历一个 __weak 指针判断, 循环将 referrer 指针设置成nil!!!
for (size_t i = 0; i < count; ++i) {
objc_object **referrer = referrers[i];
if (referrer) {
if (*referrer == referent) { // 判断 __weak 指针指向的是否是引用对象!!!
*referrer = nil;
} else if (*referrer) {
_objc_inform("__weak variable at %p holds %p instead of %p. "
"This is probably incorrect use of "
"objc_storeWeak() and objc_loadWeak(). "
"Break on objc_weak_error to debug.\n",
referrer, (void*)*referrer, (void*)referent);
objc_weak_error();
}
}
}
// 将这个 entry 从 weak_table中移除
weak_entry_remove(weak_table, entry);
}
源码中的逻辑非常清楚, 大概流程如下:
- release以后, 对象如果需要释放, 就会调用
dealloc - 判断对象的isa, 按照顺序释放关联对象, cxx析构函数
- 判断弱引用标志位, 处理weak_table中的弱引用指针, 然后将对象关联的weak_entry从弱引用表中移除
- 判断是否有使用引用计数表, 处理附加引用计数表!!