[在Swift中使用标准C库函数时,我遇到了问题在传递C字符串时。作为一个简单示例(仅用于演示问题),标准C库函数
char * strdup(const char *s1);
暴露给Swift为
func strdup(_: CString) -> UnsafePointer<CChar>
这意味着strdup()
的返回值不能传递给另一个strdup()
调用:
let s1 : CString = "abc"
let s2 = strdup(s1) // OK, s2 is a UnsafePointer<CChar>
let s3 = strdup(s2) // error: could not find an overload for '__conversion' that accepts the supplied arguments
我的问题是:如何从CString
创建Swift UnsafePointer<CChar>
,这样一个标准库函数返回的C字符串可以传递给另一个函数?
我唯一可以找到的方法是(使用How do you convert a String to a CString in the Swift Language?中的代码):
let s2a = String.fromCString(s2).bridgeToObjectiveC().UTF8String
let s3 = strdup(s2a)
但是由于两个原因,我认为这并不令人满意:
备注/背景:当然,使用诸如Swift String
或Objective-C NSString
之类的高级数据结构的高级函数是更可取的。但是其中有BSD功能标准C库,在Foundation框架中没有完全对应的库。
我在尝试回答Accessing temp directory in Swift时遇到了这个问题。这里,mkdtemp()
是一个BSD函数,不存在确切的NSFileManager
替换项(我所知道的)。mkdtemp()
返回一个UnsafePointer<CChar>
,必须将其传递给NSFileManager
功能stringWithFileSystemRepresentation
,需要CString
论点。
更新:从Xcode 6 beta 6开始,此问题不再存在,因为已简化了C字符串到Swift的映射。你可以写
let s1 = "abc" // String
let s2 = strdup(s1) // UnsafeMutablePointer<Int8>
let s3 = strdup(s2) // UnsafeMutablePointer<Int8>
let s4 = String.fromCString(s3) // String
Swift 1.1(或更早的版本)具有更好的C字符串桥接:
let haystack = "This is a simple string"
let needle = "simple"
let result = String.fromCString(strstr(haystack, needle))
CString
类型已完全消失。
Swift中的String结构具有一个初始化例程,您可以像这样使用它:
let myString = String(cString: myUnsafePointer)