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

标签 c swift interop unsafe-pointers

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

相关文章:

c++ - 我想知道我们使用的ide是否包含链接器

animation - AppDelegate 中的动画后不显示 Root View Controller

ios - 通过在 Swift 中按下按钮触发本地通知权限对话框

c# - 将 C# 字符串作为参数发送到非托管 C++ DLL 函数

c# - 从 C# 传递位值

使用指针和数组创建随机故事生成器来处理字符串

c - C程序执行时间

c - 为什么使用常量被认为比使用文字更好的编程风格?

ios - 数据未在 View Controller Swift 之间传输

c# - 使用 C# 对 Excel 中的列进行排序