問題
-all_load 是在Objective-C 編譯時常用到的一個參數,比如這篇文章所介紹的,生成靜態庫的一些問題-all_load。但是我們在加入這個參數后,有時會出現“ld: duplicate symbol _main“的錯誤,比如 libCurl duplicate symbol,但是把 -all_load 參數從 Other Like Flags 里去掉后,編譯雖然可以通過,但在運行時會崩潰。問題出在什么地方呢?
分析
參考 Universal static library problem in iPhone SDK(需翻牆,通過bing的cache查看),cocos2d for Mac,Building Objective-C static libraries with categories 等文章,我們可以大概知道 -all_load 的作用:
Objective C 中特有的語法特性 Category 大家肯定很熟悉,類似於C#中的擴展方法,可以在一個類的外面擴展這個類的功能,使得我們可以方便的為系統類添加自己的功能,比如為 NSString 添加 md5 編碼。在編譯到靜態庫時,這些代碼模塊實際上是存在不同的obj文件里的。程序在連接Category方法時,實際上只加載了Category模塊,擴展的基類代碼並沒有被加載。這樣,程序雖然可以編譯通過,但是在運行時,因為找不到基類模塊,就會出現 unrecognized selector 這樣的錯誤。
但是使用了 -all_load 之后,隨之而來的問題就是,有的庫里自作主張的定義了main函數,比如 libcurl.a。誰知道他們定義main函數是干什么的,但問題就是這個樣子。因此,-all_load,加還是不加,這是個問題。
解決方案
有兩個解決方案,其實都是尋找 -all_load 的替代參數。
蘋果的官方文檔里給了我們一個解決方案,使用 -ObjC 參數。它的文檔說,-ObjC 參數會把所有的 Objective-C 代碼模塊加載,所以程序會有點臃腫;也許libcurl不是 Objective-C 實現的,所以不會加載。但相對於巨大的圖片聲音資源來說,程序稍微變大一點應該不算什么。
另外一個解決方案是 使用 -force_load 參數,可以參考 Objective-C categories in static library,How can I avoid “duplicate symbol” errors in xcode with shared static libraries?。我們可以只加載感興趣的 靜態庫。
感謝神奇的 StackOverflow!
