在GoLang中使用嵌套结构打印结构的奇怪行为

简介:

通常在尝试打印 结构时 ,我们会使用%v来显示结构的所有数据。它将打印结构中每个字段的默认格式。

 
%v默认格式的值
打印结构时,加号标志(%+ v)添加字段名称

但是最近我们在使用嵌套结构打印结构时发现了一个奇怪的行为,该结构具有一个 String()字符串%v 格式根据我们的理解打印出“意外”输出。

我们先看看示例代码段。

 
package main import ( "fmt" ) type Inner struct { } type A struct {
Inner
FieldA string}func (i Inner) String() string { return "anything"}func main() {
myA := A{FieldA: "A"}
fmt.Printf("%v", myA)}

我们期望产量

{任何东西A}

但实际结果是

什么

这没有意义,对吧?这似乎 FIELDA 如果被忽略 的字符串()字符串 是为实现 内部 结构类型。

根据我们的理解,struct A类型有两个字段: InnerFieldA,在打印值时,它将循环遍历字段并使用其默认格式进行打印。所以Inner应该调用它的String(),而FieldA将打印它的字符串值。但是,上述输出让我们怀疑我们的理解。

仔细研究一下这些文档之后,它就有了以下规则。

 ●  如果格式(对于Println等隐式%v)对字符串有效(%s%q%v%x%X),则适用以下两个规则:

 ●  如果操作数实现了错误接口,则将调用Error方法将对象转换为字符串,然后根据动词的需要对其进行格式化(如果有)。

如果操作数实现方法String()字符串,则将调用该方法将对象转换为字符串,然后根据动词的需要对其进行格式化(如果有)。

由于 Inner 是一个操作数,它实现了 String()字符串 方法,该方法被视为 Stringer 接口,因此在打印时会调用它。这解释了我们实际看到的输出。下面实际上是Go中源代码的一部分。

 
switch verb { case 'v' , 's' , 'x' , 'X' , 'q' : // Is it an error or Stringer? // The duplication in the bodies is necessary: // setting handled and deferring catchPanic // must happen before calling the method. switch v := p . arg . ( type ) { case error :
handled = true defer p.catchPanic(p.arg, verb)
p.fmtString(v.Error(), verb) return case Stringer:
handled = true defer p.catchPanic(p.arg, verb)
p.fmtString(v.String(), verb) return }}

现在如果我们在A中有两个嵌套结构并且它们都实现了 String()字符串 怎么办?

 
package main import ( "fmt" ) type Inner struct { } type InnerAgain struct { } type A struct {
Inner
InnerAgain
FieldA string}func (i Inner) String() string { return "anything"}func (i InnerAgain) String() string { return "nothing"}func main() {
myA := A{FieldA: "A"}
fmt.Printf("%v", myA)}

输出是我们最初的预期

{什么都不是啊}

上面输出的原因是调用String()是不明确的,因此它将回退遍历所有字段并以默认格式获取它们的值。

请注意Go提供的上述行为,以便在使用%v格式在日志中打印struct时不会错过一些重要数据。


原文发布时间为:2018-11-11

本文来自云栖社区合作伙伴“Golang语言社区”,了解相关信息可以关注“Golang语言社区”。

相关文章
|
5月前
|
编译器 Go
Golang:Go语言结构
Go 语言结构 在我们开始学习 Go 编程语言的基础构建模块前,让我们先来了解 Go 语言最简单程序的结构。
80 0
|
6月前
|
存储 编译器 Go
100天精通Golang(基础入门篇)——第8天:Go语言程序的流程结构和条件语句
100天精通Golang(基础入门篇)——第8天:Go语言程序的流程结构和条件语句
34 0
|
编译器 Go 索引
巨细靡遗流程控制,Go lang1.18入门精炼教程,由白丁入鸿儒,Go lang流程结构详解EP09
流程结构就是指程序逻辑到底怎么执行,进而言之,程序执行逻辑的顺序。众所周知,程序整体都是自上由下执行的,但有的时候,又不仅仅是从上往下执行那么简单,大体上,Go lang程序的流程控制结构一共有三种:顺序结构,选择结构,循环结构。顺序结构:从上向下,逐行执行;选择结构:条件满足,某些代码才会执行,0-1次;循环结构:条件满足,某些代码会被反复的执行多次,0-N次
巨细靡遗流程控制,Go lang1.18入门精炼教程,由白丁入鸿儒,Go lang流程结构详解EP09
|
Go Python
Golang学习之路(二):Golang的语言结构和变量
Golang学习之路(二):Golang的语言结构和变量
113 0
Golang学习之路(二):Golang的语言结构和变量
|
6月前
|
存储 编译器 Go
Golang 语言的多种变量声明方式和使用场景
Golang 语言的多种变量声明方式和使用场景
32 0
|
6月前
|
缓存 编译器 Go
Golang 语言 vendor 在 GOPATH 和 Modules 中的区别
Golang 语言 vendor 在 GOPATH 和 Modules 中的区别
31 0
|
6月前
|
Go 数据中心 微服务
Golang 语言微服务的服务发现组件 Consul 的系统架构介绍
Golang 语言微服务的服务发现组件 Consul 的系统架构介绍
59 0
|
1月前
|
SQL 前端开发 Go
编程笔记 GOLANG基础 001 为什么要学习Go语言
编程笔记 GOLANG基础 001 为什么要学习Go语言
|
6月前
|
存储 JSON Go
Golang 语言 gRPC 服务怎么同时支持 gRPC 和 HTTP 客户端调用?
Golang 语言 gRPC 服务怎么同时支持 gRPC 和 HTTP 客户端调用?
76 0
|
6月前
|
存储 安全 Go
Golang 语言微服务的服务注册与发现组件 Consul
Golang 语言微服务的服务注册与发现组件 Consul
57 0