10

我正在尝试与 Swift 的旧 C 终端应用程序进行交互。我已成功集成源代码并将标头从 C 桥接到 Swift。该代码从 Xcode 6.3 beta 编译和运行。我已将终端应用程序的主入口点重命名为:

int initialize(int argc, char **argv);

不过,我正在努力将参数从 Swift 传递给这个 C 函数。我的挑战是以正确的格式转换参数。来自 Swift 的典型输入如下所示:

let args = ["-c", "1.2.3.4", "-p", "8000"]

我试过弄乱“cStringUsingEncoding(NSUTF8StringEncoding)”和“withUnsafePointer”,但到目前为止还没有运气。任何帮助是极大的赞赏!

4

2 回答 2

12

C 函数

int initialize(int argc, char **argv);

映射到 Swift 为

func initialize(argc: Int32, argv: UnsafeMutablePointer<UnsafeMutablePointer<Int8>>) -> Int32

这是一个可能的解决方案:

let args = ["-c", "1.2.3.4", "-p", "8000"]

// Create [UnsafeMutablePointer<Int8>]:
var cargs = args.map { strdup($0) }
// Call C function:
let result = initialize(Int32(args.count), &cargs)
// Free the duplicated strings:
for ptr in cargs { free(ptr) }

它使用了在strdup($0) Swift 中字符串$0自动转换为 C 字符串的事实,如String value to UnsafePointer<UInt8> function parameter behavior中所述。

于 2015-04-06T10:38:40.837 回答
1

基于 Martin 的回答,如果您发现自己经常这样做,您可以将 dup/free 部分包装成类似于以下样式的函数String.withCString

import Darwin

func withCStrings
  <R, S: SequenceType where S.Generator.Element == String>
  (strings: S, @noescape body:  (UnsafeBufferPointer<UnsafeMutablePointer<Int8>>) -> R) 
  -> R  {

    let cstrings = map(strings) { strdup($0) } + [nil]

    let result = cstrings.withUnsafeBufferPointer(body)

    for ptr in cstrings { free(ptr) }

    return result
}

let execvargs = ["/usr/bin/say"] + dropFirst(Process.arguments)

let execvresult = withCStrings(execvargs) {
    execv($0[0], $0.baseAddress)
}
于 2015-04-06T11:59:26.200 回答