在 Go 中,指针对于处理变量至关重要。然而,理解不同指针类型之间的细微差别可能具有挑战性。本文探讨了一个特定的场景,展示了指针之间的区别以及在使用 fmt 包中的默认格式时它们如何影响输出。
以下代码片段演示了问题:
type Test struct { Test string } var Models = map[string]interface{}{ "test": newTest(), } func main() { test1 := Test{} fmt.Println("Test 1: ") fmt.Printf("%v", test1) fmt.Println() fmt.Println("Test 1 as pointer: ") fmt.Printf("%v", &test1) fmt.Println() test2 := Models["test"] fmt.Println("Test 2: ") fmt.Printf("%v", test2) fmt.Println() fmt.Println("Test 2 as pointer: ") fmt.Printf("%v", &test2) } func newTest() Test { var model Test return model }
执行代码时,您会注意到输出有所不同将 test2 与 test1 作为指针打印。 test1 作为指针的输出是空字符串,而 test2 作为指针的输出是地址的十六进制表示形式。
fmt.Printf 函数使用 %v默认格式的动词,它根据要打印的值的类型选择特定格式。对于指针,默认格式是地址的十六进制表示形式。
在第一种情况(test1 作为指针)中,打印的值是指向 Test 结构的指针。但是,由于该结构体是用零值初始化的,因此输出为空。
在第二种情况(test2 作为指针)中,打印的值是指向 interface{} 的指针。在到达 %v 动词之前,该值在另一个 interface{} 内进行了额外的包装,然后指向 Models["test"]。由于最终打印的值是指向 interface{} 的指针,因此将应用指针的默认格式,并且您将获得地址的十六进制表示形式。
要解决此问题,需要使用类型断言从 test2 中提取实际的 Test 结构。这可以通过以下方式实现:
t2 := Models["test"] test2 := t2.(Test) // test2 is now of type Test
通过类型断言,test2 变量现在指向 Test 结构,您可以像 test1 一样打印它。
或者,可以使用通过在映射中存储 *Test 值直接指向 Test 的指针:
var Models = map[string]*Test{ "test": newTest(), }
这种方法消除了类型断言和包装的需要在interface{}中,从而避免了地址的十六进制表示。
以上是Go 的'fmt.Printf”和指针类型如何交互,以及为什么它们针对不同的指针类型产生不同的输出?的详细内容。更多信息请关注PHP中文网其他相关文章!