简述
人们的生活方式已经发生了很大的变化,这是因为互联网的在近些年发展的非常迅速。
互联网在未来很多年都是一个热门的发展项目,但是这个行业的竞争力也非常大,需要不断的创新。但是学习总有一个流程与方案#
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