一尘不染

字符串值到UnsafePointer 功能参数行为

swift

我发现以下代码可以编译并运行:

func foo(p:UnsafePointer<UInt8>) {
    var p = p
    for p; p.memory != 0; p++ {
        print(String(format:"%2X", p.memory))
    }
}

let str:String = "今日"
foo(str)

这将打印出来E4BB8AE697A5,并且是UTF8的有效表示形式今日

据我所知,这是未记录的行为。从文档中:

当一个函数被声明为带有UnsafePointer参数时,它可以接受以下任何一项:

  • nil,作为空指针传递
    * UnsafePointer,UnsafeMutablePointer或AutoreleasingUnsafeMutablePointer值,必要时将其转换为UnsafePointer
  • 一个In-out表达式,其操作数是Type类型的左值,作为左值的地址传递
  • 一个[Type]值,该值作为指向数组开头的指针传递,并在调用期间进行生命周期扩展

在这种情况下,str不是他们。

我想念什么吗?


添加:

如果参数类型是 UnsafePointer<UInt16>

func foo(p:UnsafePointer<UInt16>) {
    var p = p
    for p; p.memory != 0; p++ {
        print(String(format:"%4X", p.memory))
    }
}
let str:String = "今日"
foo(str)
//  ^ 'String' is not convertible to 'UnsafePointer<UInt16>'

即使内部String表示形式为UTF16

let str = "今日"
var p = UnsafePointer<UInt16>(str._core._baseAddress)
for p; p.memory != 0; p++ {
    print(String(format:"%4X", p.memory)) // prints 4ECA65E5 which is UTF16 今日
}

阅读 615

收藏
2020-07-07

共1个答案

一尘不染

之所以如此,是因为Swift团队自首次发布以来就进行了互操作性更改-
您说对了,看来它还没有纳入文档范围。StringUnsafePointer<UInt8>需要的地方工作,因此您可以调用需要const char *参数的C函数,而无需进行大量额外工作。

查看strlen在“ shims.h”中定义的C函数:

size_t strlen(const char *s);

在Swift中,它是这样实现的:

func strlen(s: UnsafePointer<Int8>) -> UInt

String无需额外的工作即可调用:

let str = "Hi."
strlen(str)
// 3
2020-07-07