iOS开发---ObjC标志

在做一个项目的时候,需要使用到一个第三方库,这个库的使用向导里面特别说明,在添加完该库后,需要在Xcode的Build Settings下Other Linker Flags里面加入-ObjC标志。

之所以使用该标志,和Objective-C的一个重要特性:类别(category)有关。根据这里的解释,Unix的标准静态库实现和Objective-C的动态特性之间有一些冲突:Objective-C没有为每个函数(或者方法)定义链接符号,它只为每个类创建链接符号。这样当在一个静态库中使用类别来扩展已有类的时候,链接器不知道如何把类原有的方法和类别中的方法整合起来,就会导致你调用类别中的方法时,出现"selector not recognized",也就是找不到方法定义的错误。为了解决这个问题,引入了-ObjC标志,它的作用就是将静态库中所有的和对象相关的文件都加载进来。

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

明白了这些,我就清楚,这个第三方库中一定是定义了一些类别来扩着已有类,一些第三方库里对系统库的类加了 category 这时,就需要使用编译参数: -ObjC ,这样第三方库中对系统类作的扩展方法才能在工程中使用。

但是使用 -Objc 后,会产生两个问题:

1 、 undefined symbols 

2 、 duplicate symbol

undefined symbols 说明工程中没有引入第三方库引用的库。

如果不加 -ObjC 参数,则在编译时不会出错,但运行时用到了没引用库的方法,则系统会崩溃。

这个问题好解决。只要找到缺的库,引入工程就可以了。

duplicate symbol 的问题就不好解决了,说明第三方库中的类名和工程中的类名,或其它第三方库中的类名重名了。

最常见的比如 Reachability 这个类,如果第三方库中引用了,自己的工程中也引用了,就会报 duplicate symbol 。

如果引用的第三方库没有源代码,就只能将自己工程中的重名类重命名了。

应对上面的问题,如果自己要制作一个库的话,要注意两点:

1 、避免对系统类加 category, 这样,别人用你的库时,不加 ObjC 参数也可以用你的库。

2 、如果库中用到了其它的第三方的源代码,尤其是用的比较普遍的,如 Reachability, 一定一定要对

这些类重命名,最常见的作法就是给类名加个前缀。以避免别人用你的库时,产生 duplicate symbol 的问题。


-ObjC

这个flag告诉链接器把库中定义的Objective-C类和Category都加载进来。这样编译之后的app会变大(因为加载了其他的objc代码进来)。但是如果静态库中有类和category的话只有加入这个flag才行。

-all_load

这个flag是专门处理-ObjC的一个bug的。用了-ObjC以后,如果类库中只有category没有类的时候这些category还是加载不进来。变通方法就是加入-all_load或者-force-load。-all_load会强制链接器把目标文件都加载进来,即使没有objc代码。-force_load在xcode3.2后可用。但是-force_load后面必须跟一个只想静态库的路径。

你可能感兴趣的:(iOS开发---ObjC标志)