1、首先,我們要先明白 other linker flag 的作用,它的意思是連接器,至于連接器是什么那就要先了解程序運行的步驟:
源文件經過一系列處理以后,會生成對應的.obj文件,然后一個項目必然會有許多.obj文件,并且這些文件之間會有各種各樣的聯系,例如函數調用。鏈接器做的事就是把這些目標文件和所用的一些庫鏈接在一起形成一個完整的可執行文件。
蘋果官方給出的解釋是:
翻譯過來的意思:大概意思就是Objective-C的鏈接器并不會為每個方法建立符號表,而是僅僅為類建立了符號表。這樣的話,如果靜態庫中定義了已存在的一個類的分類,鏈接器就會以為這個類已經存在,不會把分類和核心類的代碼合起來。這樣的話,在最后的可執行文件中,就會缺少分類里的代碼,這樣函數調用就失敗了。
解決辦法:
在Other Linker Flags里加上所需的參數,用到的參數一般有以下3個:
(1)-ObjC:加了這個參數后,鏈接器就會把靜態庫中所有的Objective-C類和分類都加載到最后的可執行文件中,雖然這樣可能會因為加載了很多不必要的文件而導致可執行文件變大,但是這個參數很好地解決了我們所遇到的問題。但是事實真的是這樣的嗎?如果-ObjC參數真的這么有效,那么事情就會簡單多了。
(2)-all_load:當靜態庫中只有分類而沒有類的時候,-ObjC參數就會失效了。這時候,就需要使用-all_load或者-force_load了。
-all_load會讓鏈接器把所有找到的目標文件都加載到可執行文件中,但是千萬不要隨便使用這個參數!假如你使用了不止一個靜態庫文件,然后又使用了這個參數,那么你很有可能會遇到ld: duplicate symbol錯誤,因為不同的庫文件里面可能會有相同的目標文件,所以建議在遇到-ObjC失效的情況下使用-force_load參數。
(3)-force_load所做的事情跟-all_load其實是一樣的,但是-force_load需要指定要進行全部加載的庫文件的路徑,這樣的話,你就只是完全加載了一個庫文件,不影響其余庫文件的按需加載。
(4)如果以上三種配置都沒起作用,應該是你把.m文件導入到項目中了,仔細去查找看哪個類導入錯誤了