基于动态库加载研究ObjC的+load方法

ObjC运行时库中的+load方法在使用时有一系列规则,本文从源码层面分析这些规则背后的原理。+load有如下规则:

  1. class中或者category中实现的+load方法会调用并且只调用一次。
  2. class的+load方法会在class所有的superclass的+load方法之后调用。
  3. category的+load方法会在class的+load调用之后调用。
  4. +load方法中向与自身class处于同一个动态库的其他class发送消息,其他class的+load可能没有执行。
+load方法在什么时候调用?

程序在开始运行时dyld(动态库加载器)会加载系统动态库,自己使用的动态库和可执行文件,这些文件均是Mach-O文件。比如我们在一个动态库DyLibA的class中实现了+load方法。当dyld将所有Mach-O文件加载完成,包括我们的动态库,然后对所有的动态库和可执行文件进行初始化。这个流程在dyld的源码中调用流程如下所示:

// 初始化可执行文件
void initializeMainExecutable() {
    ...
    // 初始化可执行文件依赖的动态库
    sImageRoots[i]->runInitializers(gLinkContext, initializerTimes[0]);
    ...
}

void ImageLoader::runInitializers(const LinkContext& context, InitializerTimingList& timingInfo)
{
    ...
    processInitializers(context, thisThread, timingInfo, up);
    ...
}

void ImageLoader::processInitializers(const LinkContext& context, mach_port_t thisThread,
                                     InitializerTimingList& timingInfo, ImageLoader::UninitedUpwards& images)
{
    ...
    images.images[i]->recursiveInitialization(context, thisThread, images.images[i]->getPath(), timingInfo, ups);
    ...
}

void ImageLoader::recursiveInitialization(const LinkContext& context, mach_port_t this_thread, const char* pathToInitialize,
                                          InitializerTimingList& timingInfo, UninitedUpwards& uninitUps)
{
    ...

    // 让objc知道我们即将初始化这个image
    uint64_t t1 = mach_absolute_time();
    fState = dyld_image_state_dependents_initialized;
    oldState = fState;
    context.notifySingle(dyld_image_state_dependents_initialized, this, &timingInfo);
    
    // 初始化这个image
    bool hasInitializers = this->doInitialization(context);

    // 告知其他人这个image已经初始化完成
    fState = dyld_image_state_initialized;
    oldState = fState;
    context.notifySingle(dyld_image_state_initialized, this, NULL);

    ...
}

recursiveInitialization方法中对动态库的初始化是根据动态库的依赖来进行的,也就是初始化当前动态库时,当前动态库所依赖的其他动态库都已经初始化完成。在recursiveInitialization方法中要对一个库进行初始化时在context.notifySingle()方法中执行objc注册给dyld的回调函数load_images,而在load_images中就会执行我们实现的+load方法。objc在_objc_init()方法中向dyld注册了回调函数,如下所示:

void _objc_init(void)
{
    static bool initialized = false;
    if (initialized) return;
    initialized = true;
    
    // fixme defer initialization until an objc-using image is found?
    environ_init();
    tls_init();
    static_init();
    lock_init();
    exception_init();
    
    // objc向dyld注册回调函数
    _dyld_objc_notify_register(&map_images, load_images, unmap_image);
}

_dyld_objc_notify_register(&map_images, load_images, unmap_image)中的第一个参数map_images中会调用realizeClass()创建类对象,
中第二个参数load_images中就会调用我们现实的+load方法。

所以+load方法是在+load方法所在的动态库初始化的时候调用的。从上述源码流程来看,+load方法只会调用一次,从而解释了第一条规则:class中或者category中实现的+load方法会调用并且只调用一次

+load在动态库内部的调用情况

ObjC注册给dyld的回调函数load_image的源码如下所示:

void
load_images(const char *path __unused, const struct mach_header *mh)
{
    // Return without taking locks if there are no +load methods here.
    if (!hasLoadMethods((const headerType *)mh)) return;

    recursive_mutex_locker_t lock(loadMethodLock);

    // Discover load methods
    {
        rwlock_writer_t lock2(runtimeLock);
        prepare_load_methods((const headerType *)mh);
    }

    // Call +load methods (without runtimeLock - re-entrant)
    call_load_methods();
}

在其中,通过prepare_load_methods((const headerType *)mh)查找+load方法,然后通过call_load_methods()调用查找到的+load方法。

void prepare_load_methods(const headerType *mhdr)
{
    size_t count, i;

    runtimeLock.assertWriting();
    
    // 获取non-lazy class
    classref_t *classlist = 
        _getObjc2NonlazyClassList(mhdr, &count);
    
    for (i = 0; i < count; i++) {
        // 递归,先获取class的super class的+load方法,然后获取自己的+load方法
        schedule_class_load(remapClass(classlist[i]));
    }
    
    // 获取实现了+load的category
    category_t **categorylist = _getObjc2NonlazyCategoryList(mhdr, &count);
    for (i = 0; i < count; i++) {
        category_t *cat = categorylist[i];
        Class cls = remapClass(cat->cls);
        if (!cls) continue;  // category for ignored weak-linked class
        realizeClass(cls);
        assert(cls->ISA()->isRealized());
        // 获取category的+load方法
        add_category_to_loadable_list(cat);
    }
}

在prepare_load_methods中,通过_getObjc2NonlazyClassList获取non-lazy class,non-lazy class就是实现了+load方法的class,通过方法schedule_class_load将class的+load方法依据父类在前子类在后保存进了loadable_classes数组;通过_getObjc2NonlazyCategoryList获取了non-lazy category,最终category的+load保存进了loadable_categories数组。

  • non-lazy class
    实现了+load方法的class是non-lazy class,在ObjC向dyld注册回调的方法中:"_dyld_objc_notify_register(&map_images, load_images, unmap_image)",第一个回调map_images中就会对non-lazy class进执行realizeClass(cls)【对class进行第一次初始化】。而通过打断来看,回调map_imags是在回调load_images之前执行的,也就是说在执行+load方法时,对应的class都是经过第一次初始化的。
  • lazy class
    源码中或者注释中并没有具体地给出lazy class的定义,我们就把除了non-lazy class之外的class看做是lazy class。这类class的realizeClass(cls)是放在对该类发送消息时,如果该class没有经过初始化,那么会对这个class执行realizeClass(cls)
void call_load_methods(void)
{
    static bool loading = NO;
    bool more_categories;

    loadMethodLock.assertLocked();

    // Re-entrant calls do nothing; the outermost call will finish the job.
    if (loading) return;
    loading = YES;

    void *pool = objc_autoreleasePoolPush();

    do {
        // 1. Repeatedly call class +loads until there aren't any more
        // 调用loadable_classes中所有的+load方法
        while (loadable_classes_used > 0) {
            call_class_loads();
        }

        // 2. Call category +loads ONCE
        // 调用loadable_categories中所有的+load方法
        more_categories = call_category_loads();

        // 3. Run more +loads if there are classes OR more untried categories
    } while (loadable_classes_used > 0  ||  more_categories);

    objc_autoreleasePoolPop(pool);

    loading = NO;
}

通过call_load_methods()调用查找到的+load方法,从代码中可以看到先通过call_class_load()调用了loadable_classes中保存的所有的class的+load方法,然后才通过call_category_loads()调用了loadable_categories中保存的所有的category的+load方法。

所以规则2:class的+load方法会在class所有的superclass的+load方法之后调用和规则3:category的+load方法会在class的+load调用之后调用得到了保证。

对于规则4:+load方法中向与自身class处于同一个动态库的其他class发送消息,其他class的+load可能没有执行;通过call_load_methods方法,可以看到同一个动态库内的+load方法都是顺序执行,所以一个class的+load方法内使用到了动态库内部的另一个class,然而另一个class的+load方法是否执行了我们并不确定,所以使用另一个class的最终结果是不确定的。

但是如果当前动态库A中的+load方法中使用了另外一个动态库B中的class,那么动态库B一定会先于动态库A进行初始化,所以动态库B中的所有+load方法已经执行完毕,所以动态库A初始化时执行的+load方法中用到动态库B中的类都是完整初始化过的,不存在不确定的结果。

总结

首先从动态库加载的角度明确了+load方法执行的时机,是在+load所在的动态库/可执行文件初始化的时候执行的;在执行一个动态库的+load方法时,当前库依赖的其他动态库已经完成了初始化,也就是其相应的+load方法已经执行完成。

其次,从ObjC源码的角度明确了父类的+load方法先于子类的+load方法执行,class的+load方法先于category的+load方法执行。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容