標籤:
現象
最近項目組在做一些第三方功能的整合,不止一次的遇到第三方庫衝突的問題,報錯如下:
| 123456 |
duplicate symbol _OBJC_METACLASS_$_JKSerializer in: /Users/tony/Desktop/XXXProject/Lib/libMiPushSDK.a(JSONKit.o) /Users/tony/Library/Developer/Xcode/DerivedData/XXXProject-boqkajmzatzxohbyrrhklfiuknic/Build/Products/Debug-iphoneos/libPods.a(JSONKit.o)ld: 24 duplicate symbols for architecture armv7clang: error: linker command failed with exit code 1 (use -v to see invocation) |
在這個報錯中,原因是第三方中自己打包了JSONKit庫,而我們的項目中也使用了這個庫。這種情況需要我們將第三方SDK中衝突的庫移除掉,即將它拆包後重打包處理。
解決步驟
首先按照上述錯誤中提到的路徑找到庫檔案:/Users/tony/Desktop/XXXProject/Lib/libMiPushSDK.a,將它拷貝一份出來做進一步處理。
| 12 |
cd~/&&mkdirlibrepack&&cdlibrepackcp/Users/tony/Desktop/XXXProject/Lib/libMiPushSDK.a./libx.a |
查看包資訊:lipo -info libx.a
如果提示fat file,那麼代表這個包是支援多平台的,例如armv7,armv7s,i386等,這需要我們逐一做解包重打包操作。否則我們只需要做一次[1-6]操作即可
建立臨時檔案夾,用於存放armv7平台解壓後的.o檔案:mkdir armv7
取出armv7平台的包:lipo libx.a -thin armv7 -output armv7/libx-armv7.a
查看庫中所包含的檔案清單:ar -t armv7/libx-armv7.a
解壓出object file(即.o尾碼檔案):cd armv7 && ar xv libx-armv7.a
找到衝突的包(JSONKit),刪除掉rm JSONKit.o
重新打包object file:cd .. && ar rcs libx-armv7.a armv7/*.o,可以再次使用[2]中命令確認是否已成功將檔案去除
將其他幾個平台(armv7s, i386)包逐一做上述[1-6]操作
重新合并為fat file的.a檔案:
lipo -create libx-armv7.a libx-armv7s.a libx-i386.a -output libMiPushSDK-new.a
拷貝到項目中覆蓋源檔案:
cp libMiPushSDK-new.a /Users/tony/Desktop/XXXProject/Lib/libMiPushSDK.a
iOS編程:第三方靜態庫(.a檔案)處理
在引用第三方庫時,不時的會碰到諸如庫衝突、庫包含了某些禁用的API等問題,而這些庫往往都被打包成了靜態庫檔案(即 .a檔案)來使用。這時就需要我們能夠去對Object file進行一些必要的處理調整。如檢索資訊,移除衝突的庫等。以下是一些常用的方法:
一、 尋找.a檔案中是否包含相關資訊,如蘋果禁用的unique Identifier。
find . -name *.a |xargs grep uniqueIdentifier
查看object file檔案資訊:
nm xxx.a | grep ".o:" | sed "s/.(//g" | sed "s/).//g" | uniq
二、去除不同的.a檔案之間內含衝突的庫,如openUDID庫的衝突。
1. 查看architectures
lipo -info libx.a
2. 如果是fat file,處理成non-fat
lipo libx.a -thin armv7 -output libx-armv7.a
3. 解壓出object file
ar xv libx-armv7.a
4. 打包object file
ar rcs libx.a *.o
5. 產生Fat file
lipo -create Libarmv7.a -output libx.a
6. 合并.a檔案
lipo -create Libarmv6.a Libarmv7.a -output Lib.a
iOS 第三方庫衝突的處理