- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我在将 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/
有没有办法像 withCString 这样链接函数? ?我的意思是任何 类似于 f :: Foo -> (CFoo -> IO a) -> IO a 的函数. 例如,假设有一个函数 cFunc ::
我正在使用函数String.withCString()如下: let s = "Hey!" let c = s.withCString { strlen($0) } println(c) 但是
我正在尝试在 Swift 中实现 AES 128,最近遇到了将 Swift 字符串写入 CChar 数组的问题: let testString = "myTestString" var keyPtr
将要描述的问题与我之前的问题有关: string.withCString and UnsafeMutablePointer(mutating: cstring) wrapped into a func
我在将 String.withCString{} 与 UnsafeMutablePointer(mutating: ...) 结合使用时遇到问题。 给定:像这样的 C 函数 randomSign(xm
我是一名优秀的程序员,十分优秀!