iOSiOS_UI

iOS底层探索之dyld(一):动态链接器流程分析

2021-07-13  本文已影响0人  俊而不逊

1. 前言

现在的互联网行业,是一天比一天卷,除了底层是必考点了,还有关于APP的性能优化也是面试常问的点。

太难了
在优化之前必须要对应用程序加载的流程熟悉,那么本次博文就对dyld进行底层的初步探索分析。 在这里插入图片描述

2. 程序加载原理

2.1 代码编译过程

我们都知道代码编写完成,必须通过编译器编译才能变成可以执行的文件。

程序编译流程
程序的执行,是把可执行的文件,加载到内存中去执行的,这个可执行的文件(Mach-O)的运行必须依赖很多的库(.a/.lib/.so),库是可执行的二进制文件,是能够被加载到内存中去的。这些库,可以分为静态库动态库

2.2 静态库和动态库

2.3 dyld 动态链接器

dyld是iOS操作系统的一个重要组成部分,在系统内核做好程序准备工作之后,会交由dyld负责余下的工作。
dyld的作用:加载各个库,也就是image镜像文件,由dyld从内存中读到表中,加载主程序,link链接各个动静态库,进行主程序的初始化工作。

程序加载流程

上面这个图是dyld的加载工作流程图,图中简单的描述了动态库注册和动态库的加载过程,具体分析还得去看底层源码,那么接下来就去探索分析。

3. 初识dyld

3.1 寻找dyld入口

程序的执行我们都知道是从main函数开始的,那么dyld在程序的那个阶段执行的呢?是在main函数执行前,还是再main函数执行后呢?这我也不得而知。

啊!你这个博主,奇奇怪怪的,你写的博文,你会不知道啊?

信你个鬼
这个嘛,就得探索探索了!首先建一个工程将main.m改写如下:
__attribute__((constructor)) void JPFunc(){
     printf("来了老弟 : %s \n",__func__);
}

int main(int argc, char * argv[]) {
    NSString * appDelegateClassName;
    @autoreleasepool {
        NSLog(@"这是main函数打印");
        // Setup code that might create autoreleased objects goes here.
        appDelegateClassName = NSStringFromClass([AppDelegate class]);
    }
    return UIApplicationMain(argc, argv, nil, appDelegateClassName);
}

然后运行程序,打印结果如下:

来了老弟 : JPFunc 
dyld初探[37212:516752] 这是main函数打印

这个__attribute__((constructor))是在main函数之前执行的一个函数。

main函数执行之前确实是可以执行其他函数的,那么dyld到现在好像还没有相关线索,那么继续往下探索。

main函数打上断点

断点断在main函数上

断点断在main函数上,发现在main之前还调用了一个start方法。

libdyld.dylib`start

点开start是一个libdyld.dylib start,突然想起网络上很流行的一句话,欢迎来到德莱联盟libdyld.dylib和这发音好像,哈哈!

但是通过对start下符号断点,断不住它。说明这不是入口的地方,我们知道还有一个方法+load,这个是在main之前必会调用的方法,那么就可以在Viewcontroller的写下+load方法添加断点,运行程序。在控制台输入指令bt,查看调用堆栈信息:

堆栈信息

堆栈信息是一个栈结构,先进后出,所以最底下打印的就是最先执行的。所以现在我们已经找到dyld的入口了。

3.2 获取dyld源码

_dyld_start,那么这就涉及到底层源码了,去苹果开放的源码官网opensource看看dyld源码

dyld源码

我们研究源码,就得去看最新的苹果源码,毕竟技术更新迭代很快,最新的才是最流行的,也是最香的,研究起来才有味道,dyld最新的版本是dyld-852,这部分源码是不能编译的,但是并不能妨碍我们去探索它。那么我们现在就去打开dyld这个牛逼的源码工程一探究竟吧!

3.3 初探dyld 源码

_dyld_start源码

全局搜索_dyld_start,发现又是汇编,是不是要疯了啊!
莫慌靓仔,稳住,不会汇编没有关系,请耐心往下看!

dyldbootstrap::start

在汇编里面发现了一个重要方法,dyldbootstrap::start,从红框中的注释可以知道,会调用dyldbootstrap::start这个C++函数,那么就可以去全局搜索下,看看C++函数的命名空间。

namespace dyldbootstrap

从命名空间里面,我们可以找到start

在这里插入图片描述
start函数里面返回的是dyld::_main,这就nice了,突然就很熟悉,很亲切。

博文篇幅有限,下篇继续探索。。。。。
iOS底层探索之dyld(下):动态链接器流程源码分析

4. 总结

更多内容持续更新

🌹 喜欢就点个赞吧👍🌹

🌹 觉得学习到了的,可以来一波,收藏+关注,评论 + 转发,以免你下次找不到我😁 🌹

🌹欢迎大家留言交流,批评指正,互相学习😁,提升自我🌹

上一篇 下一篇

猜你喜欢

热点阅读