我在将 String.withCString{} 与 UnsafeMutablePointer(mutating: ...) 结合使用时遇到问题。
给定:像这样的 C 函数
randomSign(xml_string: UnsafeMutablePointer<Int8>!) -> Uint
同时给出:一个像
这样的字符串str = "some xml_tag"
我的工作代码是这样的(VERSION_1)
func randomSignWrapper_1(xml_tag_str: String) -> Uint {
let result = xml_str.withCString { s in return
randomSign(UnsafeMutablePointer(mutating: s))
}
return result
}
但我希望将 withCString 放入一个单独的函数中,如下所示:
func testFunction(_ x: String) -> UnsafeMutablePointer<Int8>{
return x.withCString{s in
return UnsafeMutablePointer(mutating: s)
}
}
这样我就可以轻松地重用它 (VERSION_2)
func randomSignWrapper_2(xml_tag_str: String) -> Uint {
let result = randomSign(testFunction(xml_tag_str))
return result
}
但问题是 VERSION_1 提供了正确的返回值,而 VERSION_2 不知何故无法正常工作,并告诉我数据有误。 我想知道为什么它会这样? 以及如何解决我可以按照描述的方式使用它?
最佳答案
withCString { ... }
创建 Swift 的临时表示
字符串为以 null 结尾的 UTF-8 序列,仅在内部有效
关闭。将指针传递到闭包外部是
未定义的行为。
另请注意,您不能使用该指针改变字符串。
路过UnsafeMutablePointer(mutating: $0)
只使编译器
相信指向的内存是可变的,但实际上在做
未定义的行为也是如此,甚至可能会崩溃。
如果randomSign()
函数不修改给定的
string 那么最好的解决方案是将其 C 声明更改为
取一个常量字符串:
unsigned long randomSign(const char *xml_string);
这将被导入到 Swift 中作为
func randomSign(_ xml_string: UnsafePointer<Int8>!) -> UInt
现在您可以将 Swift 字符串直接传递给该函数:
let str = "some xml_tag"
let result = randomSign(str)
编译器会自动插入代码来创建一个 临时 UTF-8 表示并将其传递给函数,比较 String value to UnsafePointer<UInt8> function parameter behavior .
如果您不能更改 C 声明,那么您仍然可以调用它
作为罢工>
let str = "some xml_tag"
let result = randomSign(UnsafeMutablePointer(mutating: str))
也不需要您自己的辅助函数。
更新:最后的建议不正确。由创建的临时字符串
编译器仅在调用 UnsafeMutablePointer()
期间
但在 randomSign()
的调用期间不一定仍然有效.
关于c - string.withCString 和 UnsafeMutablePointer(变异 : cstring) wrapped into a function,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44026115/