首页 >后端开发 >Golang >为什么'fmt.Println”不使用结构体的'String()”方法,除非导出结构体和方法?

为什么'fmt.Println”不使用结构体的'String()”方法,除非导出结构体和方法?

Barbara Streisand
Barbara Streisand原创
2024-11-19 17:50:031098浏览

Why doesn't `fmt.Println` use a struct's `String()` method unless the struct and method are exported?

fmt.Println 和 String() 方法

为什么 fmt.Println 在调用时不利用成员的 String() 方法一个结构?让我们检查一下底层机制。

考虑以下代码:

import "fmt"

type Bar struct{}

func (b Bar) String() string {
    return "bar"
}

type Foo struct {
    B []*Bar
    BB *Bar
}

func main() {
    f := Foo{B: []*Bar{&Bar{}}, BB: &Bar{}}
    fmt.Println(f, f.B, f.BB)
}

此代码产生预期的输出:

{[(addr: 0x8201dde620)] (addr: 0x8201dde620)} [bar] bar

但是,您可能想知道为什么输出不是如下所示:

{[bar] bar} [bar] bar

它将显示成员的 String() 表示形式而不是内存地址。

原因在于 Bar 类型及其未导出的性质String() 方法。在 Go 中,未导出的成员和方法只能在定义它们的包内访问。由于 fmt.Println 不是同一包的一部分,因此它无法访问这些未导出的元素。因此,它采用打印非内置类型的内存地址的默认行为。

要解决此问题,您需要导出 Bar 类型及其 String() 方法。此外,还必须导出 Foo 结构中的字段以进行正确的 JSON 编码和解码。

这是代码的修改版本,演示了正确的方法:

import "fmt"

type Bar struct{}

func (b Bar) String() string {
    return "bar"
}

type Foo struct {
    B []*Bar
    BB *Bar
}

func main() {
    f := Foo{B: []*Bar{&Bar{}}, BB: &Bar{}}
    fmt.Println(f)
}

进行了这些更改实施后,输出现在将是:

{[bar] bar} [bar] bar

此示例强调了使用导出类型和方法与 fmt.Println 等格式化程序进行适当集成的重要性。

以上是为什么'fmt.Println”不使用结构体的'String()”方法,除非导出结构体和方法?的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn