1. 前言

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

在优化之前必须要对应用程序加载的流程熟悉,那么本次博文就对dyld进行底层的初步探索分析。

2. 程序加载原理

2.1 代码编译过程

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

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

2.2 静态库和动态库

  • 静态库: 例如.a.framework。静态库链接时,会被完整地复制到可执行文件中,被使用到了多次,就会复制多份,这样就有多份拷贝很冗余,编译时间长了,还浪费了内存空间。
  • 动态库:例如.dylib.framework。动态库链接时,只会存在一份,并不会复制多份,在内存中共享这一份,系统只加载一次,谁有用到了就去找这一份,减少了程序的体积大小,可以节省编译时间和内存空间。

    静态库都好理解,那么动态库在程序中是怎么加载到内存呢?系统是通过怎样的方式来链接的呢?这就用到了一个工具,也就是博文开头提到的dyld(the dynamic link editor)动态链接器,

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函数之前执行的一个函数。

  • 补充:GNU C的一大特色就是__attribute__机制。__attribute__可以设置函数属性(Function Attribute)、变量属性(Variable Attribute )和类型属性(Type Attribute)。

  • __attribute__书写特征是:__attribute__前后都有两个下划线,并切后面会紧跟一对原括弧,括弧里面是相应的__attribute__参数。

  • __attribute__语法格式为:__attribute__ ((attribute-list))

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

main函数打上断点

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

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

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

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

3.2 获取dyld源码

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

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

3.3 初探dyld 源码

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

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

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


start函数里面返回的是dyld::_main,这就nice了,突然就很熟悉,很亲切。

博文篇幅有限,下篇继续探索。。。。。
敬请期待!

4. 总结

  • 应用程序加载到内存中运行,会通过dyld链接各种库
  • 库分为动态库静态库
  • 静态库有多份拷贝,增加包的大小,程序加载链接时间长,浪费了内存空间。
  • 动态库存在一份,并不会复制多份,节省程序加载链接时间和内存空间
  • iOS的可执行的文件是Mach-O
  • dyld源码可以去苹果的开源opensource查看

更多内容持续更新

iOS底层探索之dyld(上):动态链接器流程分析相关推荐

  1. iOS 底层探索 - 消息转发

    一.动态方法解析流程分析 我们在上一章<消息查找>分析到了动态方法解析,为了更好的掌握具体的流程,我们接下来直接进行源码追踪. 我们先来到 _class_resolveMethod 方法, ...

  2. iOS底层原理之dyld应用程序加载

    前言 众所周知,main作为程序的入口,但是在它之前发生了什么?有点好奇,让我们来瞅一瞅: 一.准备工作 1.代码 __attribute__((constructor)) void Func(){p ...

  3. iOS底层探索(二) - 写给小白看的Clang编译过程原理

    iOS底层探索(一) - 从零开始认识Clang与LLVM 写在前面 编译器是属于底层知识,在日常开发中少有涉及,但在我的印象中,越接近底层是越需要编程基本功,也是越复杂的.但要想提升技术却始终绕不开 ...

  4. iOS底层探索二(OC 中 alloc 方法 初探)

    前言 相关文章: iOS底层探索一(底层探索方法) iOS底层探索三(内存对齐与calloc分析) iOS底层探索四(isa初探-联合体,位域,内存优化) iOS底层探索五(isa与类的关系) iOS ...

  5. iOS 底层探索篇 —— KVC 底层原理

    iOS 底层探索篇 -- KVC 底层原理 1. Method Swizzling的坑与应用 1.1 method-swizzling 是什么? 1.2 坑点 坑点1:method-swizzling ...

  6. Linux中ELF格式 可执行文件+动态链接器 的加载

    两种加载方式 (1)加载可执行文件,通过PT_INTERP加载动态链接器 (2)直接加载动态链接器,再由其加载可执行文件 On a typical ELF system such as Linux, ...

  7. linux动态可执行文件,Linux中ELF格式 可执行文件+动态链接器 的加载

    两种加载方式 (1)加载可执行文件,通过PT_INTERP加载动态链接器 (2)直接加载动态链接器,再由其加载可执行文件 On a typical ELF system such as Linux, ...

  8. cmake中添加引用动态链接_macos - CMake:MacOS上动态链接资源的运行时错误(dyld:未加载库) - SO中文参考 - www.soinside.com...

    问题 在MacOS上,我在运行时为依赖于动态链接资源的CMake项目获取链接问题 - 但仅在安装项目之后!当我只构建二进制文件而不安装它时,不会发生这个问题. $ ./testapp Hello wo ...

  9. Mach-O 的动态链接(Lazy Bind 机制)

    ➠更多技术干货请戳:听云博客 动态链接 要解决空间浪费和更新困难这两个问题最简单的方法就是把程序的模块相互分割开来,形成独立的文件,而不再将它们静态的链接在一起.简单地讲,就是不对那些组成程序的目标文 ...

  10. 链接器:绑定符号到地址上

    目录 为什么是编译器 LLVM和LLD LLVM 的编译​​​​​​​的几个主要过程 链接器做了什么 符号链接 动态库链接 实操演示动态库链接​​​​​​​         链接器最主要的作用,就是将 ...

最新文章

  1. 疯狂Spring Cloud连载(5)Eureka集群搭建
  2. mysqldump 和 sql命令导入sql文件
  3. 利用XShell上传、下载文件(使用sz与rz命令),超实用!
  4. android系统存储路径在哪里,Android 手机存储目录
  5. python用http协议传数据_《Python网络爬虫》1.3 HTTP协议基础知识
  6. Attach and Detach in VC
  7. Dockerfile 数据卷最佳实践
  8. 基于动态规划DTW算法加速衡量两个不同的时间序列的相似性
  9. http下载文件(常用方式+支持在线打开方式)
  10. matlab安时积分法计算soc,一种带加权的安时积分的SOC估算方法与流程
  11. 中基鸿业如何投资理财
  12. 一文带你了解Unity Shader-小飞侠轻功(径向模糊)
  13. 0day安全:软件漏洞分析技术(第2版)pdf
  14. Java小程序 个人缴税
  15. GPS 入门 8 —— GPS定位基本原理浅析
  16. 北京大学可视化发展前沿研究生暑期学校Day5
  17. 15、JVM监控及诊断工具-GUI篇
  18. 一步一步做项目(3)创建Web项目
  19. JS中script标签defer和async属性的区别
  20. MOSFET常用品牌及特点

热门文章

  1. 2018年11月26日 练习3
  2. 洛谷 P2473 [SCOI2008]奖励关 解题报告
  3. 使用python下载一些链接的软件包
  4. lucene 分词实现
  5. spring_150904_hibernatetemplate
  6. eclipse 配色方案
  7. Python覆盖率分析工具_Coverage
  8. C# 基于MySQL的数据层基类(MySQLHelper)
  9. 使用 Python 和 Flask 实现 RESTful services
  10. angularjs指令参数transclude