有没有办法调用从Swift的dylib加载的C函数?
这是我的 dylib 文件:
cppdemofile.cpp
#include "cppdemofile.h"
int add(int a, int b) {
return a + b;
}
cppdemofile.h
#ifndef __CppDemoLibrary__cppdemofile__
#define __CppDemoLibrary__cppdemofile__
#pragma GCC visibility push(default)
extern "C" int add(int a, int b);
#pragma GCC visibility pop
#endif
编译到dylib并检查:
nm -gU libCppDemoLibrary.dylib
0000000000000f80 T _add
...将libCppDemoLibrary.dylib
复制到~/lib
...
Swift程序:
@IBAction func buttonClick(sender: NSButton) {
let handle = dlopen("libCppDemoLibrary.dylib", RTLD_NOW)
if (handle != nil) {
var sym = dlsym(handle, "add")
if (sym != nil) {
let pointer = UnsafeMutablePointer<(CInt, CInt) -> CInt>(sym)
// When debugging, I'm reaching up to this point...
// but now, how do I call the 'add' function here???
// var result = ???
// label.stringValue = "Total: " + String(result)
}
}
}
如何调用add
功能?使用dylib可以吗?我应该将这些来源添加到我的快速项目吗?
答案 0 :(得分:11)
因为你,可以从Swift调用add
函数
将其定义为与extern "C"
建立C链接。
使库成为Swift模块(如上面的jtbandes所示)
评论)可能是更好的解决方案,
但是这里是如何使用来自Swift的dlsym()
返回的函数指针:
首先添加
typedef int(*addFunc)(int, int);
到桥接头文件,或者定义
typealias addFunc = @convention(c) (CInt, CInt) -> CInt
在斯威夫特。然后以下工作:
let handle = dlopen(path, RTLD_NOW)
if (handle != nil) {
var sym = dlsym(handle, "add")
if (sym != nil) {
let f = unsafeBitCast(sym, addFunc.self)
let result = f(12, 45)
print(result)
}
dlclose(handle)
}
如果addFunc
与if let handle = dlopen(path, RTLD_NOW) {
if let sym = dlsym(handle, "add") {
let f = unsafeBitCast(sym, to: addFunc.self)
let result = f(12, 45)
print(result)
}
dlclose(handle)
}
不符,那么这会崩溃
加载函数的实际签名。
更新 Swift 3:
{{1}}
答案 1 :(得分:0)
为了在Swift中使用C ++代码,您需要将它包装在C函数或Objective-C类中。