关于 Xcode 上的 Other Linker Flags
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-CClass
和Category
都加载到最后的可执行文件中 -
-all_load
会让链接器把所有找到的目标文件都加载到可执行文件中,但是千万不要随便使用这个参数!假如你使用了不止一个静态库文件,然后又使用了这个参数,那么你很有可能会遇到ld: duplicate symbol
错误,因为不同的库文件里面可能会有相同的目标文件,所以建议在遇到-ObjC
失效的情况下使用-force_load
参数。 -
-force_load
所做的事情跟-all_load
其实是一样的,但是-force_load
需要指定要进行全部加载的库文件的路径,这样的话,你就只是完全加载了一个库文件,不影响其余库文件的按需加载
之所以使用该标志,和 Objective-C 的一个重要特性:类别(Category)有关。根据这里的解释,Unix 的标准静态库实现和 Objective-C 的动态特性之间有一些冲突:Objective-C 没有为每个 Function
(或者Method
)定义链接符号,它只为每个 Class
创建链接符号。这样当在一个静态库中使用 Category
来扩展已有类的时候,链接器不知道如何把 Class
原有的 Method
和 Category
中的 Method
整合起来,就会导致你调用 Category
中的 Method
时,出现 selector not recognized
,也就是找不到方法定义的错误。为了解决这个问题,引入了 -ObjC
标志,它的作用就是将静态库中所有的和对象相关的文件都加载进来。
本来这样就可以解决问题了,不过在 64 位的 Mac 系统或者 iOS 系统下,链接器有一个 bug,会导致只包含有类别的静态库无法使用 -ObjC
标志来加载文件。变通方法是使用 -all_load
或者 -force_load
标志,它们的作用都是加载静态库中所有文件,不过 all_load
作用于所有的库,而 -force_load
后面必须要指定具体的文件。