面试

iOS dealloc底层深入(dealloc做了什么,为什么弱

2020-12-17  本文已影响0人  iOS劝退师

源码:objc4-723,地址:https://opensource.apple.com/tarballs/objc4/
调用流程:首先调用 _objc_rootDealloc() -> 接下来调用 rootDealloc() -> object_dispose()-> clearDeallocating()

一、调用 _objc_rootDealloc()

// Replaced by NSZombies
- (void)dealloc {
    _objc_rootDealloc(self);
}

二、调用 rootDealloc()

void
_objc_rootDealloc(id obj)
{
    assert(obj);

    obj->rootDealloc();
}

inline void
objc_object::rootDealloc()
{
    if (isTaggedPointer()) return;  // fixme necessary?

    if (fastpath(isa.nonpointer  &&  //优化过isa指针
                 !isa.weakly_referenced  &&  //不存在弱引用指向
                 !isa.has_assoc  &&  //没设置过关联对象
                 !isa.has_cxx_dtor  && // 没有c++的析构函数(.cxx_destruct)
                 !isa.has_sidetable_rc))// 不存在引用计数器是否过大无法存储在isa中(使用 sidetable 来存储引用计数)
    {
        assert(!sidetable_present());
        free(this);
    } 
    else {
        object_dispose((id)this);
    }
}

2.1、首先判断 isTaggedPointer 是否是标记指针 是直接 return ;
注:Tagged Pointer技术,用于优化NSNumber、NSDate、NSString等小对象的存储(https://www.jianshu.com/p/70551a5e77c0
2.2、其次判断该对象是否可以被快速释放。一共有5个判断依据:
nonpointer 是否优化过isa指针(类似Tagger Pointer)
weakly_reference 是否存在弱引用指向
has_assoc 是否设置过关联对象
has_cxx_dtor 是否有c++的析构函数(.cxx_destruct)
has_sidetable_rc 引用计数器是否过大无法存储在isa中(使用 sidetable 来存储引用计数)

三、 object_dispose()
如果不能快速释放,则调用 object_dispose()方法,做下一步的处理(调用objc_destructInstance)

/***********************************************************************
* object_dispose
* fixme
* Locking: none
**********************************************************************/
id 
object_dispose(id obj)
{
    if (!obj) return nil;

    objc_destructInstance(obj);    
    free(obj);

    return nil;
}
/***********************************************************************
* objc_destructInstance
* Destroys an instance without freeing memory. 
* Calls C++ destructors.
* Calls ARC ivar cleanup.
* Removes associative references.
* Returns `obj`. Does nothing if `obj` is 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);//(Calls C++ destructors.)c++的销毁器来销毁成员变量
        if (assoc) _object_remove_assocations(obj);//(Calls ARC ivar cleanup.)用来释放动态绑定的对象
        obj->clearDeallocating();
    }

    return obj;
}

3.1、判断是否存在c++的析构函数,有则调用object_cxxDestruct()

3.1.1、object_cxxDestruct()

object_cxxDestruct这个方法最终会调用objc_storeStrong来释放成员变量(实例变量)

3.2、移除关联对象_object_remove_assocations(常用于category中添加带变量的属性),关联对象定义:https://www.cnblogs.com/limicheng/p/4186214.html

3.3、调用clearDeallocating()方法

inline 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());
}

sidetable_clearDeallocating和clearDeallocating_slow,并最终都调用:weak_clear_no_lock,该方法将所有指向所提供对象的所有弱指针置清空。

总结:clearDeallocating一共做了两件事
将对象弱引用表清空,即将弱引用该对象的指针置为nil
清空引用计数表(当一个对象的引用计数值过大(超过255)时,引用计数会存储在一个叫 SideTable 的属性中,此时isa的 has_sidetable_rc 值为1),这就是为什么弱引用不会导致循环引用的原因

上一篇下一篇

猜你喜欢

热点阅读