简述
人们的生活方式已经发生了很大的变化,这是因为互联网的在近些年发展的非常迅速。
互联网在未来很多年都是一个热门的发展项目,但是这个行业的竞争力也非常大,需要不断的创新。但是学习总有一个流程与方案#
dyld的分析:
程序的加载原理图——(代码编写完成,必须通过编译器编译才能变成可以执行的文件)

那么动态库在程序中是怎么加载到内存呢?系统是通过怎样的方式来链接的呢?这就用到了一个工具,那就是动态链接器!
动态链接器

以上为加载工作流程图,动态库的注册过程
分析底层源码——dyld入口
这个嘛,就得探索探索了!首先建一个工程将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
链接:https://juejin.cn/post/6984338883815145502
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
【腾讯文档】更多资料分享
https://docs.qq.com/doc/DZXpKSU5LdEdja0hR