现象
最近项目组在做一些第三方功能的集成,不止一次的遇到第三方库冲突的问题,报错如下:
1
2
3
4
5
6
|
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 armv7
clang: error: linker command failed with exit code 1 (use -v to see invocation)
|
在这个报错中,原因是第三方中自己打包了JSONKit库,而我们的项目中也使用了这个库。这种情况需要我们将第三方SDK中冲突的库移除掉,即将它拆包后重打包处理。
解决步骤
首先按照上述错误中提到的路径找到库文件:/Users/tony/Desktop/XXXProject/Lib/libMiPushSDK.a
,将它拷贝一份出来做进一步处理。
1
2
|
cd~/&&mkdirlibrepack&&cdlibrepack
cp/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.a2. 如果是fat file,处理成non-fat
lipo libx.a -thin armv7 -output libx-armv7.a3. 解压出object file
ar xv libx-armv7.a4. 打包object file
ar rcs libx.a *.o5. 生成Fat file
lipo -create Libarmv7.a -output libx.a6. 合并.a文件
lipo -create Libarmv6.a Libarmv7.a -output Lib.a