阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

简单介绍Golang打印复杂结构体的两种方法

307次阅读
没有评论

共计 3017 个字符,预计需要花费 8 分钟才能阅读完成。

导读 在 Golang 语言开发中,我们经常会使用结构体类型,如果我们使用的结构体类型的变量包含指针类型的字段,我们在记录日志的时候,指针类型的字段的值是指针地址,将会给我们 debug 代码造成不便
fmt 结构体占位符

在 Golang 中有原生的 fmt 格式化工具去打印结构体,可以通过占位符 %v、%+v、%#v 去实现,这 3 种的区别如下所示:

type User struct {
    Name string
    Age  int
}
func main() {
    user := User{
        Name: "张三",
        Age:  95,
    }
    fmt.Printf("%v\n", user)
    fmt.Printf("%+v\n", user)
    fmt.Printf("%#v\n", user)
}

打印结果如下所示:

{张三 95}                     
{Name: 张三 Age:95}            
main.User{Name:"张三", Age:95}

其中的区别:

  • %v 占位符是不会打印结构体字段名称的,字段之间以空格隔开;
  • %+ v 占位符会打印字段名称,字段之间也是以空格隔开;
  • %#v 占位符则会打印结构体类型和字段名称,字段之间以逗号分隔
  • 打印复杂结构体

    当结构体中的字段是指针类型时,用占位符直接打印出来的是怎样的呢?

    还是以前面的例子为基础,我们给“张三”加一条狗,其中 User 结构体中引入的 Dog 是指针类型,代码如下:

    type Dog struct {
        Name string
        Age  int
    }
    type User struct {
        Name string
        Age  int
        Dog  *Dog
    }
    func main() {
        dog := Dog{
            Name: "旺财",
            Age:  2,
        }
        user := User{
            Name: "张三",
            Age:  95,
            Dog:  &dog,
        }
        fmt.Println(user)
        fmt.Printf("%v\n", user)
        fmt.Printf("%+v\n", user)
        fmt.Printf("%#v\n", user)
    }

    这时还能把所有值打印出来吗?

    {张三 95 0xc000004078}                                       
    {Name: 张三 Age:95 Dog:0xc000004078}                          
    main.User{Name:"张三", Age:95, Dog:(*main.Dog)(0xc000004078)}

    这时可以看到 Dog 字段打印的不是 Dog 结构体内部的值,而是一个地址值。很显然,这个不是我们需要在日志中看到的,我们需要看的是结构体具体的值,那这个值又怎么打印呢?

    方案一

    实现 String() 或 GoString() 方法

    Golang 中的 fmt 包中有一个 Stringer 接口,接口中只有一个 String() 方法

    // Stringer is implemented by any value that has a String method,
    // which defines the ``native'' format for that value.
    // The String method is used to print values passed as an operand
    // to any format that accepts a string or to an unformatted printer
    // such as Print.
    type Stringer interface {String() string
    }

    我们可以让 User 和 Dog 结构体分别实现 String() 方法,这种方法类似于 Java 中的 toString() 方法。基于前面的代码,我们增加如下 String() 方法实现:

    func (d *Dog) String() string {return "{\"name"+"\": \"" + d.Name + "\","+"\""+"age\": \"" + strconv.Itoa(d.Age) + "\"}"
    }
    func (u *User) String() string {return "{\"name"+"\": \"" + u.Name + "\", \""+"age\": \"" + strconv.Itoa(u.Age) + "\", \"dog\": "+ u.Dog.String() +"}"
    }

    运行后,打印的结果如下所示:

    {张三 95 {"name": "旺财","age": "2"}}                        
    {Name: 张三 Age:95 Dog:{"name": "旺财","age": "2"}}           
    main.User{Name:"张三", Age:95, Dog:(*main.Dog)(0xc000004078)}

    发现,实现 String() 方法只对 %v 和 %+v 占位符有效,对于 %#v 占位符,其打印的结构体指针类型还是一个地址值。

    其实在 fmt 包中,Stringer 接口 下面,我们还可以看到另外一个 GoStringer 接口:

    // GoStringer is implemented by any value that has a GoString method,
    // which defines the Go syntax for that value.
    // The GoString method is used to print values passed as an operand
    // to a %#v format.
    type GoStringer interface {GoString() string
    }
    The GoString method is used to print values passed as an operand to a %#v format. (GoString 方法用于打印作为操作数传递给 %#v 格式的值)

    找到了,我们再实现 GoString() 方法,就可以用 %#v 占位符打印结构体指针类型中的值了。

    基于之前代码增加如下代码:

    func (d *Dog) GoString() string {return "{\"name"+"\": \"" + d.Name + "\","+"\""+"age\": \"" + strconv.Itoa(d.Age) + "\"}"
    }
    func (u *User) GoString() string {return "{\"name"+"\": \"" + u.Name + "\", \""+"age\": \"" + strconv.Itoa(u.Age) + "\", \"dog\": "+ u.Dog.String() +"}"
    }

    运行后,打印结果如下所示,这下子就都可以打印了:

    {张三 95 {"name": "旺财","age": "2"}}                          
    {Name: 张三 Age:95 Dog:{"name": "旺财","age": "2"}}             
    main.User{Name:"张三", Age:95, Dog:{"name": "旺财","age": "2"}}

    到这里,我感觉这种方案有点麻烦呢,还有没有其他不用维护 String() 或 GoString() 的方法呢?

    方案二

    转换成 json 格式

    func main() {
        dog := Dog{
            Name: "旺财",
            Age:  2,
        }
        user := User{
            Name: "张三",
            Age:  95,
            Dog:  &dog,
        }
        byteUser, _ := json.Marshal(&user)
        fmt.Println(string(byteUser))
    }

    打印结果如下所示,如果使用 json 库的话,是可以直接把结构体指针类型的具体值都打印出来的,比较方便:

    {"Name":"张三","Age":95,"Dog":{"Name":"旺财","Age":2}}

    到此这篇关于 Golang 打印复杂结构体两种方法详解的文章就介绍到这了

    阿里云 2 核 2G 服务器 3M 带宽 61 元 1 年,有高配

    腾讯云新客低至 82 元 / 年,老客户 99 元 / 年

    代金券:在阿里云专用满减优惠券

    正文完
    星哥玩云-微信公众号
    post-qrcode
     0
    星锅
    版权声明:本站原创文章,由 星锅 于2024-07-24发表,共计3017字。
    转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
    【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
    阿里云-最新活动爆款每日限量供应
    评论(没有评论)
    验证码
    【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中

    星哥玩云

    星哥玩云
    星哥玩云
    分享互联网知识
    用户数
    4
    文章数
    19348
    评论数
    4
    阅读量
    7802902
    文章搜索
    热门文章
    开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

    开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

    开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南 大家好,我是星哥。之前介绍了腾讯云的 Code...
    星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

    星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

    星哥带你玩飞牛 NAS-6:抖音视频同步工具,视频下载自动下载保存 前言 各位玩 NAS 的朋友好,我是星哥!...
    云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

    云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

    云服务器部署服务器面板 1Panel:小白轻松构建 Web 服务与面板加固指南 哈喽,我是星哥,经常有人问我不...
    我把用了20年的360安全卫士卸载了

    我把用了20年的360安全卫士卸载了

    我把用了 20 年的 360 安全卫士卸载了 是的,正如标题你看到的。 原因 偷摸安装自家的软件 莫名其妙安装...
    星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

    星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

    星哥带你玩飞牛 NAS-3:安装飞牛 NAS 后的很有必要的操作 前言 如果你已经有了飞牛 NAS 系统,之前...
    阿里云CDN
    阿里云CDN-提高用户访问的响应速度和成功率
    随机文章
    星哥带你玩飞牛NAS-13:自动追番、订阅下载 + 刮削,动漫党彻底解放双手!

    星哥带你玩飞牛NAS-13:自动追番、订阅下载 + 刮削,动漫党彻底解放双手!

    星哥带你玩飞牛 NAS-13:自动追番、订阅下载 + 刮削,动漫党彻底解放双手! 作为动漫爱好者,你是否还在为...
    告别Notion焦虑!这款全平台开源加密笔记神器,让你的隐私真正“上锁”

    告别Notion焦虑!这款全平台开源加密笔记神器,让你的隐私真正“上锁”

      告别 Notion 焦虑!这款全平台开源加密笔记神器,让你的隐私真正“上锁” 引言 在数字笔记工...
    如何免费使用强大的Nano Banana Pro?附赠邪修的用法

    如何免费使用强大的Nano Banana Pro?附赠邪修的用法

    如何免费使用强大的 Nano Banana Pro?附赠邪修的用法 前言 大家好,我是星哥,今天来介绍谷歌的 ...
    星哥带你玩飞牛NAS-14:解锁公网自由!Lucky功能工具安装使用保姆级教程

    星哥带你玩飞牛NAS-14:解锁公网自由!Lucky功能工具安装使用保姆级教程

    星哥带你玩飞牛 NAS-14:解锁公网自由!Lucky 功能工具安装使用保姆级教程 作为 NAS 玩家,咱们最...
    使用1Panel面板搭建属于你的AI项目环境

    使用1Panel面板搭建属于你的AI项目环境

    使用 1Panel 面板搭建属于你的 AI 项目环境 在 AI 项目越来越火的今天,很多朋友都想自己动手搭建一...

    免费图片视频管理工具让灵感库告别混乱

    一言一句话
    -「
    手气不错
    支付宝、淘宝、闲鱼又双叕崩了,Cloudflare也瘫了连监控都挂,根因藏在哪?

    支付宝、淘宝、闲鱼又双叕崩了,Cloudflare也瘫了连监控都挂,根因藏在哪?

    支付宝、淘宝、闲鱼又双叕崩了,Cloudflare 也瘫了连监控都挂,根因藏在哪? 最近两天的互联网堪称“故障...
    让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级

    让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级

    让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级 大家好,我是星哥,之前写了一篇文章 自己手撸一...
    开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

    开源MoneyPrinterTurbo 利用AI大模型,一键生成高清短视频!

      开源 MoneyPrinterTurbo 利用 AI 大模型,一键生成高清短视频! 在短视频内容...
    星哥带你玩飞牛NAS-7:手把手教你免费内网穿透-Cloudflare tunnel

    星哥带你玩飞牛NAS-7:手把手教你免费内网穿透-Cloudflare tunnel

    星哥带你玩飞牛 NAS-7:手把手教你免费内网穿透 -Cloudflare tunnel 前言 大家好,我是星...
    国产开源公众号AI知识库 Agent:突破未认证号限制,一键搞定自动回复,重构运营效率

    国产开源公众号AI知识库 Agent:突破未认证号限制,一键搞定自动回复,重构运营效率

    国产开源公众号 AI 知识库 Agent:突破未认证号限制,一键搞定自动回复,重构运营效率 大家好,我是星哥,...