首页 > 后端开发 > Golang > Go 的'fmt.Printf”和指针类型如何交互,以及为什么它们针对不同的指针类型产生不同的输出?

Go 的'fmt.Printf”和指针类型如何交互,以及为什么它们针对不同的指针类型产生不同的输出?

Linda Hamilton
发布: 2024-12-09 20:21:18
原创
596 人浏览过

How Do Go's `fmt.Printf` and Pointer Types Interact, and Why Do They Produce Different Outputs for Different Pointer Types?

评估 Go 指针的差异

在 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中文网其他相关文章!

来源:php.cn
本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
作者最新文章
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板