5

在 macOS 上,我使用必须由用户安装的外部框架(用 C 编写)。在 Swift 中,我需要在运行时检查它是否存在,并且我不能使用 #available() 因为它用于与操作系统相关的功能,并且我正在尝试追踪外部框架。此外, NSClassFromString() 没有用,因为它不是一个 Objective-C 框架。

我一直在尝试了解如何复制 Objective-C 等效项以检查弱链接符号,例如:

if ( anExternalFunction == NULL ) {
    // fail graciously
} else {
    // do my thing
}

但在 Swift 中,这似乎不起作用:编译器指出,由于 anExternalFunction 不是可选的,我将始终得到 != nil,这使“Swift 有意义”,但对我没有一点帮助。

我找到了两种解决方案,但它们让我的代码变得很臭,就像你不会相信的那样:

选项 1,使用名为 isFrameworkAvailable() 的函数创建一个 Objective-C 文件,并从 Swift 调用

选项 2,实际使用以下 Swift 代码检查库:

let libHandle = dlopen("/Library/Frameworks/TheLib.framework/TheLib", RTLD_NOW)

if (libHandle != nil) {
    if dlsym(libHandle, "anExternalFunction") != nil {
        return true
    }
}
return false

我无法让选项 2 与 RTLD_DEFAULT 很好地配合使用,因为由于某种原因,它在 dlfcn.h (-2) 中定义,但似乎没有在 Swift 中导入(就像该标头中的所有负指针:RTLD_NEXT、RTLD_DEFAULT ,RTLD_SELF,RTLD_MAIN_ONLY)。我发现这个最丑陋的黑客使它起作用:

if dlsym(unsafeBitCast(-2, to: UnsafeMutableRawPointer.self), "anExternalFunction") != nil {
    // library installed
}

因此,对于选项 2,我需要路径或 hack,我发现它特别难看(但它有效),选项 1 不是很“Swifty”,对我来说没有什么意义:在 Swift 中这样做的正确方法是什么?

编辑:澄清问题,更好地解释选项 2。

4

1 回答 1

0

我发现了两种方法:

    1.
if let _ = dlopen("/Library/Frameworks/MyLibrary.framework/MyLibrary", RTLD_NOW) {
  print("Can open my library")
}
    2.
#if canImport(MyLibrary)
  print("Can import my library")
#endif

从可读性和类型安全的角度来看,第二个显然更好,但我不确定两者之间的权衡是什么。

于 2019-06-26T22:36:03.103 回答