关于 Xcode 上的 Other Linker Flags

关于 Xcode 上的 Other Linker Flags_第1张图片

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 ClassCategory 都加载到最后的可执行文件中

  • -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 原有的 MethodCategory 中的 Method 整合起来,就会导致你调用 Category 中的 Method 时,出现 selector not recognized,也就是找不到方法定义的错误。为了解决这个问题,引入了 -ObjC 标志,它的作用就是将静态库中所有的和对象相关的文件都加载进来。

本来这样就可以解决问题了,不过在 64 位的 Mac 系统或者 iOS 系统下,链接器有一个 bug,会导致只包含有类别的静态库无法使用 -ObjC 标志来加载文件。变通方法是使用 -all_load 或者 -force_load 标志,它们的作用都是加载静态库中所有文件,不过 all_load 作用于所有的库,而 -force_load 后面必须要指定具体的文件。

你可能感兴趣的:(关于 Xcode 上的 Other Linker Flags)