ios程序編譯鏈接參數 all_load 的 ld duplicate symbol _main 的 bug及修復


問題

-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 MacBuilding 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 libraryHow can I avoid “duplicate symbol” errors in xcode with shared static libraries?。我們可以只加載感興趣的 靜態庫。

 

感謝神奇的 StackOverflow


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM