如何在iOS上使用嵌入式专用框架和应用程序中的静态库

时间:2015-09-09 16:43:08

标签: ios objective-c xcode frameworks linker

在私有框架,应用程序和扩展中使用静态库中的类的正确方法是什么?我的示例项目可以在https://github.com/keithyipkw/framework

找到

在第二次提交中,SDK与.a链接。运行应用程序创建了错误

Ld /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/FrameworkApp.app/FrameworkApp normal x86_64
    cd /Users/keithyip/Documents/Workspace/FrameworkApp
    export IPHONEOS_DEPLOYMENT_TARGET=8.4
    export PATH="/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/usr/bin:/Applications/Xcode.app/Contents/Developer/usr/bin:/usr/bin:/bin:/usr/sbin:/sbin"
    /Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/clang -arch x86_64 -isysroot /Applications/Xcode.app/Contents/Developer/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator8.4.sdk -L/Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator -F/Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator -filelist /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Intermediates/FrameworkApp.build/Debug-iphonesimulator/FrameworkApp.build/Objects-normal/x86_64/FrameworkApp.LinkFileList -Xlinker -rpath -Xlinker @executable_path/Frameworks -Xlinker -objc_abi_version -Xlinker 2 -fobjc-arc -fobjc-link-runtime -Xlinker -no_implicit_dylibs -mios-simulator-version-min=8.4 /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/SDK.framework/SDK -Xlinker -dependency_info -Xlinker /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Intermediates/FrameworkApp.build/Debug-iphonesimulator/FrameworkApp.build/Objects-normal/x86_64/FrameworkApp_dependency_info.dat -o /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/FrameworkApp.app/FrameworkApp

Undefined symbols for architecture x86_64:
  "_OBJC_CLASS_$_GAI", referenced from:
      objc-class-ref in AppDelegate.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)

符号在.a中是全局的,但在SDK中是本地的

$ nm -a /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/SDK.framework/SDK | grep '_OBJC_CLASS_$_GAI'
00000000000d94e0 s _OBJC_CLASS_$_GAI

在第三次提交中,我将.a添加到了应用目标。该应用程序运行但有警告

objc[3743]: Class GAI is implemented in both /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/SDK.framework/SDK and /Users/keithyip/Library/Developer/CoreSimulator/Devices/752A7B8E-405E-4403-BDD8-A168613774B1/data/Containers/Bundle/Application/D16B5121-2DA9-452B-9574-95B35AE3E197/FrameworkApp.app/FrameworkApp. One of the two will be used. Which one is undefined.

我根据警告

中的文件路径检查了SDK和app二进制文件
$ nm -a /Users/keithyip/Library/Developer/Xcode/DerivedData/FrameworkApp-bpzqozighjdtncegosucvgelzagc/Build/Products/Debug-iphonesimulator/SDK.framework/SDK | grep '_OBJC_CLASS_$_GAI'
00000000000d94e0 s _OBJC_CLASS_$_GAI

$ nm -a /Users/keithyip/Library/Developer/CoreSimulator/Devices/752A7B8E-405E-4403-BDD8-A168613774B1/data/Containers/Bundle/Application/D16B5121-2DA9-452B-9574-95B35AE3E197/FrameworkApp.app/FrameworkApp | grep '_OBJC_CLASS_$_GAI'
0000000100032c88 s _OBJC_CLASS_$_GAI

深入挖掘nm -m,libGoogleAnalyticsServices.a与其他静态库不同。链接到动态库中的其他库是没有问题的。

0000000000002b20 (__DATA,__objc_data) private external _OBJC_CLASS_$_GAI

似乎不可能将符号公开

clang: error: invalid argument '-keep_private_externs' not allowed with '-dynamiclib'

3 个答案:

答案 0 :(得分:4)

通常,您只需将静态库链接到动态库即可。但是,在这种情况下,.a中的符号是私有外部(visibility = hidden),不可能将libGoogleAnalyticsServices.a包含到动态库中,并使用iOS上动态库外部的符号。这个时刻解决它的选项是

  1. 将动态库更改为静态库
  2. 重构代码以从动态库中删除GA
  3. 为GA创建包装类,以便被调用者不需要链接到GA
  4. 如果将来遇到类似情况,其他选项

    1. 要求供应商公开符号
    2. 检查是否有任何" objcopy --globalize-symbols"相当于iOS。当我写这篇文章时,最接近的是objconv。它支持OS X,但不支持iOS。 An objcopy equivalent for Mac / iPhone?

答案 1 :(得分:2)

拿起你的项目并尝试了很多东西。看起来它唯一可行的就是使用-flat_namespace编译选项。默认情况下,所有内容都使用两级命名空间构建。

如果使用-flat_namespace选项,链接器将允许您使用-undefined suppress,以防止未定义符号出错。结果是,当执行最终链接并将google.a与应用版本链接时,他们都将完全解析。

不幸的是,谷歌图书馆不是扁平化版本,所以除非你可以获得平面版本,否则这个版本无法使用。

确实有效的一件事是将SDK目标更改为静态目标。把你发布的内容和单独进行的更改编译并运行时没有错误。所以这似乎是你最好的选择,或者通过你在评论中提到的其他方式来避免这个问题。

答案 2 :(得分:0)

我遇到了类似的问题,除了在OS X上:我正在创建一个带有dylib的框架,并在静态库中进行链接。我想在静态库上为使用该框架的应用程序提供公共符号。即使我在我想要导出的静态库类/方法上设置了正确的可见性属性,这些符号也被剥离从最终库中删除。

一种解决方法是在动态库中编写一些代码以在静态库中引用这些符号;这让连接器不会从静态lib中剥离导出符号。

最终解决方案最终是将静态库构建为可重定位目标文件;生成的动态库包含了我想要的静态库中的所有导出符号,并且不需要动态库中的参考代码位。