在 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。