load_images

作者: Priders | 来源:发表于2021-08-26 10:47 被阅读0次

    load_images

    load_image 会对每个image进行加载 多次调用

    if (!didInitialAttachCategories && didCallDyldNotifyRegister) {
            didInitialAttachCategories = true;
            loadAllCategories();
        }
    

    新版objc 把加载分类的方法延迟到了
    完成_dyld_objc_notify_register调用之后
    旧版在map_images时候 就已经统一加载了
    所以本章节 要对loadAllCategories 进行分析

    // Return without taking locks if there are no +load methods here.
    if (!hasLoadMethods((const headerType *)mh))
        return;
    

    判断没有load 方法直接return
    _getObjc2NonlazyClassList 快速获取 来判断
    _getObjc2NonlazyCategoryList

    prepare_load_methods

    准备加载load相关方法
    1、加载非懒加载类的load方法。

    递归加载父类load方法
    把cls加入全局+load的类列表

    2、加载分类的load方法。

    取出分类的宿主类
    realize宿主类(分类的宿主类不一定实现了load方法 因为 未实现的load的类在mapImages阶段 并不会realize)
    把cat加入全局+load的分类列表

    加入全局的load列表两者方式类似 
    取出类方法 load是类方法 遍历selName == load即可
    类 ==> ISA()->data()->ro()->baseMethods()
    分类 ==>直接取出类列表 
    

    call_load_methods

    先执行类的load方法
    在执行分类的load方法

    loadAllCategories

    加载所有的分类

    新版objc 把加载分类的方法延迟到了
    完成_dyld_objc_notify_register调用之后
    旧版在map_images时候 就已经统一加载了

    全局header_info List 遍历一遍 调用 在mapImages时机存有
    load_categories_nolock(hi);

    load_categories_nolock

    processCatlist(hi->catlist(&count));
    processCatlist(hi->catlist2(&count));
    两个匿名函数调用
    

    如果分类中有类方法还是实例方法,协议,或者属性/是否有类属性,
    走如下流程

    ///class_t->data is class_rw_t, not class_ro_t.    
    1、cls->isRealized() 说明已经经过realizeClassWithoutSwift
    //将分类的方法添加到rw中的methods中去
    调用attachCategories(cls, &lc, 1, ATTACH_EXISTING);
    
    2、如果没有realizeClassWithoutSwift
    将其加入到全局的map ExplicitInitDenseMap<Class, category_list> 当中去
    map class为key  cat_list 为value
    cat_list ==>是一个经过优化locstamped_category_t数组
    
    struct locstamped_category_t {
        category_t *cat;
        struct header_info *hi;
    };
    class category_list : nocopy_t {
        union {
            locstamped_category_t lc;
            struct {
                locstamped_category_t *array;
                // this aliases with locstamped_category_t::hi
                // which is an aliased pointer
                uint32_t is_array :  1;
                uint32_t count    : 31;
                uint32_t size     : 32;
            };
        } _u;
        ...
    

    attachCategories

    在旧版本中 先进行malloc
    method_list_t **mlists = (method_list_t **) malloc(cats->count * sizeof(*mlists));
    接着倒序遍历
    新版本818以上
    Only a few classes have more than 64 categories during launch. This uses a little stack, and avoids malloc.
    做了如此优化 启动时候 极少数的分类大于64 个 避免malloc
    大于64的先提前attachlists 把个数重置为0

    其中有一句  auto rwe = cls->data()->extAllocIfNeeded();
    获取分配空间的rwe 如果没有分配 会进行分配 也就是把ro的 迁移到rw
    

    旧版中是倒序遍历 到了新版本 如何处理呢?
    mlists[ATTACH_BUFSIZ - ++mcount] 数组的最大值开始往回写入
    最终处理
    rwe->methods.attachLists(mlists + ATTACH_BUFSIZ - mcount, mcount);
    指针数组的起始位置为 倒数第(mcount个数)个

    methods.attachLists

    内部维护这一个list_array_tt 类型的
    list_array_tt<method_t, method_list_t, method_list_t_authed_ptr>

    for (int i = oldCount - 1; i >= 0; i--)
                    newArray->lists[i + addedCount] = array()->lists[i];
                for (unsigned i = 0; i < addedCount; i++)
                    newArray->lists[i] = addedLists[i];
    

    把前面的分类方法的数组添加到前面去 (所以这里就会在方法查找的时候 宿主类为什么痛同名方法被覆盖的原因)

    相关文章

      网友评论

        本文标题:load_images

        本文链接:https://www.haomeiwen.com/subject/zknyiltx.html