解决 Xcode下使用公用静态库,出现的"Duplicate Symbol"错误


今天在添加某统计工具skd时,出现了几个问题,最初是报“i386”错误,可能是对方的静态.a文件不支持模拟器,纠正了这个错误后,又报“Duplicate Symbol”错误,刚看网上一个人的如下解释,解决了这个问题。感觉类似情况太常见了。

ios的Framework是共享动态库,不会被打包到app中,非系统Framework静态库都会被打包到app中,所以会产生"Duplicate Symbol"的错误。
在Build Settings->Other link flags中删除所有的-all_load与-force_load, XCode会很smart的去掉"Duplicate Symbol"。


iOS解决两个静态库的冲突 duplicate symbol

场景:

_base64_encode 函数的冲突


后来在网络上搜寻,删除掉 Other Linker Flag 的 -all_load 就可以解决静态库冲突的问题,

但是这样做的话,会使一些外部的静态库,使用objc扩展函数(catagory)的方法失效。例如BaiduMapApi

如果是有些库使用到了扩展函数(catagory)可以分别对这个库进行加载

使用:-force_load

-force_load BaiduMapApi/libs/Release-iphoneos/libbaidumapapi.a

(BaiduMapApi是添加到当前目录下的)

-force_load $(BUILT_PRODUCTS_DIR)/libxxx.a

(这里是直接添加静态库项目源码的做法)

使用-force_load分别进行加载还是蛮方便的,如果有些函数加入了main函数使用all_load就相当麻烦了。

的冲突

如果两个静态库冲突的结构是相同的,可以考虑将两个静态库拆分出来进行合并。


查看文件的架构有哪些

$ lipo -info libzbar.a 
   
 Architectures in the fat file: libzbar.a are: armv7 (cputype (12) cpusubtype (11)) i386



将armv7解压出来


lipo libzbar.a -thin armv7 -output libzbar-armv7.a


新建立一个文件夹出来存放解压的(.o)文件

$ mkdir armv7 
   
 $ cd armv7

将静态库中的文件解压

$ ar -x ../libzbar-armv7.a




然后将另一个静态库根据以上的步骤做一遍,然后观察连个解压的静态库中,有那些是一样的就合并在一起,不过注意的是两个静态库冲突的(.o)文件必须一致,否则也会出现错误。

合并完后进行打包了

$ libtool -static -o ../libnew-armv7.a *.o




如果像在虚拟机也使用,进行相同的步骤后,将i386的架构合并再一起就可以了。


合并静态库

$ lipo -create -output lib.a libnew-armv76.a libi386.a


今天在添加某统计工具skd时,出现了几个问题,最初是报“i386”错误,可能是对方的静态.a文件不支持模拟器,纠正了这个错误后,又报“Duplicate Symbol”错误,刚看网上一个人的如下解释,解决了这个问题。感觉类似情况太常见了。

ios的Framework是共享动态库,不会被打包到app中,非系统Framework静态库都会被打包到app中,所以会产生"Duplicate Symbol"的错误。
在Build Settings->Other link flags中删除所有的-all_load与-force_load, XCode会很smart的去掉"Duplicate Symbol"。

iOS解决两个静态库的冲突 duplicate symbol

场景:

_base64_encode 函数的冲突

后来在网络上搜寻,删除掉 Other Linker Flag 的 -all_load 就可以解决静态库冲突的问题,

但是这样做的话,会使一些外部的静态库,使用objc扩展函数(catagory)的方法失效。例如BaiduMapApi

如果是有些库使用到了扩展函数(catagory)可以分别对这个库进行加载

使用:-force_load

-force_load BaiduMapApi/libs/Release-iphoneos/libbaidumapapi.a

(BaiduMapApi是添加到当前目录下的)

-force_load $(BUILT_PRODUCTS_DIR)/libxxx.a

(这里是直接添加静态库项目源码的做法)

使用-force_load分别进行加载还是蛮方便的,如果有些函数加入了main函数使用all_load就相当麻烦了。

的冲突

如果两个静态库冲突的结构是相同的,可以考虑将两个静态库拆分出来进行合并。

查看文件的架构有哪些

$ lipo -info libzbar.a 
   
 Architectures in the fat file: libzbar.a are: armv7 (cputype (12) cpusubtype (11)) i386


将armv7解压出来

lipo libzbar.a -thin armv7 -output libzbar-armv7.a

新建立一个文件夹出来存放解压的(.o)文件

$ mkdir armv7 
   
 $ cd armv7



将静态库中的文件解压

$ ar -x ../libzbar-armv7.a




然后将另一个静态库根据以上的步骤做一遍,然后观察连个解压的静态库中,有那些是一样的就合并在一起,不过注意的是两个静态库冲突的(.o)文件必须一致,否则也会出现错误。

合并完后进行打包了

$ libtool -static -o ../libnew-armv7.a *.o



如果像在虚拟机也使用,进行相同的步骤后,将i386的架构合并再一起就可以了。

合并静态库

$ lipo -create -output lib.a libnew-armv76.a libi386.a