有时在开发中会发现无法识别拓展类别的添加方法,总是识别原文件的方法,而原文件中是没有新添方法的,于是在动态调用拓展方法时丢出了unrecognized selector sent to instance的错误。
解决方法是在Xcode的Build Settings下Other Linker Flags里面加入-ObjC标志。
之所以使用该标志,和Objective-C的一个重要特性:类别(category)有关。根据这里的解释,Unix的标准静态库实现和Objective-C的动态特性之间有一些冲突:Objective-C没有为每个函数(或者方法)定义链接符号,它只为每个类创建链接符号。这样当在一个静态库中使用类别来扩展已有类的时候,链接器不知道如何把类原有的方法和类别中的方法整合起来,就会导致你调用类别中的方法时,出现"selector not recognized",也就是找不到方法定义的错误。为了解决这个问题,引入了-ObjC标志,它的作用就是将静态库中所有的和对象相关的文件都加载进来。
-all_load和-force_load
在64位的Mac系统或者iOS系统下,链接器有一个bug,会导致只包含有类别的静态库无法使用-ObjC标志来加载文件。变通方法是使用-all_load 或者-force_load标志,它们的作用都是加载静态库中所有文件,不过all_load作用于所有的库,而-force_load后面必须要指定具体的文件。
-all_load的坑
在解决实际问题的过程中,发现加入原工程已经加入了-Objc标志,依然丢出了unrecognized selector sent to instance错误,于是使用了-all_load标志,不过亲测发现,可能是-all_load强制链接器把目标文件都加载进来的原因,工程出现了更多的莫名其妙的红色错误,例如无法识别出错文件中的sectionNumber变量,并且将“sectionNumber”自动修复为“se8ctionNumber”,没错,莫名其妙加了一个数字8。饶了好大一圈也无法解决,最终无法使用-all_load 标志。
最后的解决方法是使用-force_load,指定链接器加载自己的文件。