iOS 如何在嵌入式私有框架和应用程序中使用静态库

How to Use Static Library in Embedded Private Frameworks and App on iOS

在私有框架、应用程序和扩展程序中的静态库中使用 类 的正确方法应该是什么?我的示例项目可以在这里找到 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和应用二进制文件

$ 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

好像不能保留符号public

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

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

如果您使用 -flat_namespace 选项,linker 将允许您使用 -undefined suppress 以防止未定义符号出现错误。结果将是,当执行最终 link 并且您在 google.a 中使用应用程序构建 link 时,它们将完全解决。

不幸的是,google 库不是构建扁平化的,因此除非您可以构建扁平化构建,否则它无法工作。

绝对有效的一件事是将您的 SDK 目标更改为静态目标。采用您发布的内容并单独进行更改可以编译和运行而不会出现错误。因此,这似乎是您最好的选择或通过您在评论中指出的其他方式避免该问题。

通常你可以简单地link一个静态库到一个动态库。但是在这种情况下.a 中的符号是私有外部的(visibility=hidden),不可能将libGoogleAnalyticsServices.a 包含到动态库中并在iOS 上使用动态库之外的符号。此时解决它的选项是

  1. 把动态库改成静态库
  2. 重构代码从动态库中移除GA
  3. 为 GA 创建包装器 类,这样被调用者就不需要 link 到 GA

如果以后有人运行遇到类似的情况,其他的选择是

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

我遇到了类似的问题,除了 OS X:我正在创建一个内部带有 dylib 的框架,并在静态库中进行链接。我想让静态库上的 public 符号可供使用该框架的应用程序使用。尽管我已经在我想要导出的静态库 classes/methods 上设置了正确的可见性属性,但这些符号已从最终库中 剥离

一种解决方法是在动态库中编写一些代码以在静态库中引用这些符号;这导致链接器没有从静态库中删除导出符号。

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