在Go语言中,我们经常会使用类型转换来进行数据类型的转换。例如,将一个[]byte类型的切片转换为string类型的字符串。通常情况下,我们可以使用`string()`函数来进行类型转换,但在某些特殊情况下,这种方式会出现问题。在Go语言中,`(string)(unsafe.Pointer(&b))`的方式被称为"魔术指针"方式,用于将一个[]byte类型的切片转换为string类型的字符串。然而,该方式并不适用于bufio.Reader类型。为什么呢?让我们来解答这个问题。
我有一个文件。它有一些ip
1.1.1.0/24 1.1.2.0/24 2.2.1.0/24 2.2.2.0/24
我读取此文件进行切片,并使用 *(*string)(unsafe.pointer(&b)) 将 []byte 解析为字符串,但不起作用
func testinitiprangefromfile(t *testing.t) { filepath := "/tmp/test" file, err := os.open(filepath) if err != nil { t.errorf("failed to open ip range file:%s, err:%s", filepath, err) } reader := bufio.newreader(file) ranges := make([]string, 0) for { ip, _, err := reader.readline() if err != nil { if err == io.eof { break } logger.fatalf("failed to read ip range file, err:%s", err) } t.logf("ip:%s", *(*string)(unsafe.pointer(&ip))) ranges = append(ranges, *(*string)(unsafe.pointer(&ip))) } t.logf("%v", ranges) }
结果:
task_test.go:71: ip:1.1.1.0/24 task_test.go:71: ip:1.1.2.0/24 task_test.go:71: ip:2.2.1.0/24 task_test.go:71: ip:2.2.2.0/24 task_test.go:75: [2.2.2.0/24 1.1.2.0/24 2.2.1.0/24 2.2.2.0/24]
为什么 1.1.1.0/24 更改为 2.2.2.0/24 ?
改变
*(*string)(unsafe.Pointer(&ip))
到字符串(ip)它可以工作
因此,虽然将切片标头重新解释为字符串标头,但您所做的方式绝对是疯狂的,并且无法保证正常工作,但这只是间接导致问题的原因。 p>
真正的问题是,您保留了指向 bufio/Reader.ReadLine()
返回值的指针,但该方法的文档说“返回的缓冲区仅在下次调用 ReadLine 之前有效。”这意味着读者以后可以自由地重用该内存,这就是正在发生的事情。
当您以正确的方式进行转换时,string(ip)
,Go会将缓冲区的内容复制到新创建的字符串中,该字符串在将来仍然有效。但是,当您将切片键入双关语到字符串中时,您会保留完全相同的指针,一旦读取器重新填充其缓冲区,该指针就会停止工作。
如果您决定使用指针欺骗作为性能黑客以避免复制和分配......那就太糟糕了。无论如何,读取器界面都会强制您复制数据,既然如此,您应该只使用 string()
。
以上是为什么 *(*string)(unsafe.Pointer(&b)) 不适用于 bufio.Reader的详细内容。更多信息请关注PHP中文网其他相关文章!