-all_load & -force_load 的区别

  • 1、使用 -all_load 选项,这会导致所有的类的方法都会被载入,不管“你的程序”有没有用到,所有的,包括系统的,方法会被载入。这个方法不仅会导致你的目标程序大小增大,而且往往容易引起一些冲突,而这些冲突往往你无法解决(常常出现在你链接的各种库中)。

  • 2、使用 -force_load,这个你可以指定要载入所有方法的库,这个算一个还不错的解决方法
    此时使用此配置: 第一项为 -force_load 第二项为 libbaidumapapi.a 路径
    linking—>other linker flags
    2.1. -force_load
    2.2. $(SRCROOT)/lib/libbaidumapapi.a
    2.3. 最好在 projects 和 targets 中都配置


在Xcode工程的 Other linker flags 中添加 -ObjC 标志可以解决使用静态库中的 category 时出现 unrecognized selector 的问题。

UNIX 的静态库(xxx.a)其实就是一个目标文件(xxx.o)的集合。在C语言中,编译一个源文件时如果遇到在其他文件中定义的函数,则会留下一个 undefined symbol。在链接时会通过找到其他文件中的定义来确定这个symbol。

在 Objective-C 中,由于方法调用都是在运行期确定的,因此没有针对方法的symbol,只有针对类的。

这样在静态库中如果使用了category 扩展已有的类,编译器不知道如何将 category 和已有的类整合在一起,就会导致 unrecognized selector 问题。添加 -ObjC 标志后,编译器会把一个类相关的所有目标文件都加载,这样就解决了这个问题。但是由于这样做会使可执行文件体积变大,所以没有设为默认选项。

在64位 iOS 应用环境下,由于链接器的一个bug,在静态库中只有category 没有对应的 class 定义时,-ObjC标志会失效。这时可以使用 -all_load 强制加载所有目标文件,或者使用 -force_load 指定加载某一个包。

在 Xcode 4.2 之后,这个链接器 bug 已经被修复,因此 -all_load 和 -force_load 标志都不再需要了。在必要时添加-ObjC即可。


重点内容

Targets 选项下有 Other linker flags 的设置,用来填写 XCode 的链接器参数,如:-ObjC 、-all_load 、 -force_load 等。

还记得我们在学习C程序的时候,从C代码到可执行文件经历的步骤是:
源代码 > 预处理器 > 编译器 > 汇编器 > 机器码 > 链接器 > 可执行文件
在最后一步需要把 .o文件 和 C语言运行库链接起来,这时候需要用到ld 命令。源文件经过一系列处理以后,会生成对应的 .obj文件,然后一个项目必然会有许多 .obj 文件,并且这些文件之间会有各种各样的联系,例如函数调用。链接器做的事就是把这些目标文件和所用的一些库链接在一起形成一个完整的可执行文件。
如果要详细研究链接器做了什么,请看:http://www.dutor.net/index.php/2012/02/what-linkers-do/

那么,Other linker flags 设置的值实际上就是 ld命令执行时后面所加的参数。

下面逐个介绍 3 个常用参数:
-ObjC:加了这个参数后,链接器就会把静态库中所有的 Objective-C类和分类都加载到最后的可执行文件中 。
-all_load:会让链接器把所有找到的目标文件都加载到可执行文件中,但是千万不要随便使用这个参数!假如你使用了不止一个静态库文件,然后又使用了这个参数,那么你很有可能会遇到ld: duplicate symbol错误,因为不同的库文件里面可能会有相同的目标文件,所以建议在遇到 -ObjC 失效的情况下使用 -force_load 参数。
-force_load:所做的事情跟-all_load其实是一样的,但是 -force_load 需要指定要进行全部加载的库文件的路径,这样的话,你就只是完全加载了一个库文件,不影响其余库文件的按需加载。

你可能感兴趣的:(-all_load & -force_load 的区别)