gpt4 book ai didi

c - string.withCString 和 UnsafeMutablePointer(变异 : cstring) wrapped into a function

转载 作者:太空宇宙 更新时间:2023-11-04 04:23:23 24 4
gpt4 key购买 nike

我在将 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/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com