自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

看透 Go 對象內(nèi)部細(xì)節(jié)的神器

開發(fā) 后端
在調(diào)式 Go 程序時(shí),我們經(jīng)常想知道對象的內(nèi)部數(shù)據(jù)是什么樣了,以便掌握程序的運(yùn)行情況。

[[437722]]

本文轉(zhuǎn)載自微信公眾號「Golang技術(shù)分享」,作者機(jī)器鈴砍菜刀。轉(zhuǎn)載本文請聯(lián)系Golang技術(shù)分享公眾號。

在調(diào)式 Go 程序時(shí),我們經(jīng)常想知道對象的內(nèi)部數(shù)據(jù)是什么樣了,以便掌握程序的運(yùn)行情況。

一般有兩種做法:對于簡單的代碼測試,我們可以通過fmt包來打印一些對象信息;在稍復(fù)雜場景下,可以利用調(diào)式器來完成,例如 GDB、LLDB 和 Delve 等。

但是,這兩種做法都有不足之處。fmt包能打印的信息并不友好,尤其在結(jié)構(gòu)體中含有指針對象時(shí);通過調(diào)式器來調(diào)式程序也經(jīng)常受限于各種因素,例如遠(yuǎn)程訪問服務(wù)器。

示例

對于 fmt 包的能力短板,我們來看一個(gè)例子。

定義 instance 和 Inner 結(jié)構(gòu)體,其中 instance 的C屬性字段是 Inner 類型指針。

  1. type instance struct { 
  2.  A string 
  3.  B int 
  4.  C *Inner 
  5.  
  6. type Inner struct { 
  7.  D string 
  8.  E string 

實(shí)例化一個(gè) instance 對象ins

  1. func main() { 
  2.  ins := instance{ 
  3.   A: "AAAA"
  4.   B: 1000, 
  5.   C: &Inner
  6.    D: "DDDD"
  7.    E: "EEEE"
  8.   }, 
  9.  } 
  10.  fmt.Println(ins) 

此時(shí),我們想知道ins的內(nèi)部數(shù)據(jù)。通過fmt.Println(ins)語句得到的打印信息如下

  1. {AAAA 1000 0xc000054020} 

由于 C 字段是指針,所以打印出來的是一個(gè)地址0xc000054020,而地址背后的數(shù)據(jù)卻被隱藏了。顯然,這對程序排查非常不友好。

go-spew

go-spew 就是為了解決上述問題而生的,它為 Go 數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)了一個(gè)深度打印機(jī)。

同樣以上文代碼為例,這次使用 go-spew 進(jìn)行打印。

下載

  1. go get -u github.com/davecgh/go-spew/spew 

導(dǎo)包

  1. "github.com/davecgh/go-spew/spew" 

打印

  1. func main() { 
  2.  ins := instance{ 
  3.   A: "AAAA"
  4.   B: 1000, 
  5.   C: &Inner
  6.    D: "DDDD"
  7.    E: "EEEE"
  8.   }, 
  9.  } 
  10.  spew.Dump(ins) 

得到打印結(jié)果

  1. (main.instance) { 
  2.  A: (string) (len=4) "AAAA"
  3.  B: (int) 1000, 
  4.  C: (*main.Inner)(0xc0000ba0c0)({ 
  5.   D: (string) (len=4) "DDDD"
  6.   E: (string) (len=4) "EEEE" 
  7.  }) 

是不是非常詳細(xì)?

場景擴(kuò)展

指針數(shù)組

除了結(jié)構(gòu)體中含有指針對象時(shí)打印 fmt 打印不夠清晰,如果數(shù)組或者map中是指針對象時(shí),傳統(tǒng)的打印同樣不友好。

  1. type Demo struct { 
  2.  a int 
  3.  b string 
  4.  
  5. func main() { 
  6.  arr := [...]*Demo{{100, "Python"}, {200, "Golang"}} 
  7.  fmt.Printf("%v\n-----------------分割線-----------\n", arr) 
  8.  spew.Dump(arr) 

兩種打印的輸出結(jié)果對比

  1. [0xc00011c018 0xc00011c030] 
  2. -----------------分割線----------- 
  3. ([2]*main.Demo) (len=2 cap=2) { 
  4.  (*main.Demo)(0xc00011c018)({ 
  5.   a: (int) 100, 
  6.   b: (string) (len=6) "Python" 
  7.  }), 
  8.  (*main.Demo)(0xc00011c030)({ 
  9.   a: (int) 200, 
  10.   b: (string) (len=6) "Golang" 
  11.  }) 

孰強(qiáng)孰弱,一目了然。

循環(huán)結(jié)構(gòu)

通過 spew.Dump 方法可以將指針地址和它指向的數(shù)據(jù)都打印出來,那如果 go-spew 需要打印循環(huán)數(shù)據(jù)結(jié)構(gòu)怎么辦,它能否正確處理(而不是陷入無限循環(huán))?

定義循環(huán)結(jié)構(gòu)體對象 Circular

  1. type Circular struct { 
  2.  a    int 
  3.  next *Circular 

實(shí)例化循環(huán)結(jié)構(gòu)體對象,再分別通過 fmt 和 go-spew 進(jìn)行打印對比

  1. func main() { 
  2.  c := &Circular{1, nil} 
  3.  c.next = &Circular{2, c} 
  4.  
  5.  fmt.Printf("%+v\n----------------分割線-------------------\n", c) 
  6.  spew.Dump(c) 

得到結(jié)果

  1. &{a:1 next:0xc0000962f0} 
  2. ----------------分割線------------------- 
  3. (*main.Circular)(0xc0000962e0)({ 
  4.  a: (int) 1, 
  5.  next: (*main.Circular)(0xc0000962f0)({ 
  6.   a: (int) 2, 
  7.   next: (*main.Circular)(0xc0000962e0)(<already shown>) 
  8.  }) 
  9. }) 

再次證明 go-spew 的強(qiáng)大。

總結(jié)

go-spew 借助于 unsafe 包,為我們帶來了非常漂亮的打印功能。

當(dāng)然,go-spew 不止 Dump 方法,它也提供了其他方法,例如轉(zhuǎn)換為字符串的 Sdump 方法;輸出重定向的 Fdump 方法;與 fmt 類似的一套 Print 用法。

同時(shí),可以通過 spew.Config 進(jìn)行一些參數(shù)配置,例如設(shè)置 spew.Config.MaxDepth 用于控制打印深度。

 

調(diào)式 Go 程序時(shí),go-spew 是一個(gè)非常好用的助手工具,推薦大家使用。

 

責(zé)任編輯:武曉燕 來源: Golang技術(shù)分享
相關(guān)推薦

2011-05-25 19:31:07

Stratus信息化

2023-08-14 08:34:14

GolangHttp

2010-10-08 10:26:45

JavaScript內(nèi)

2013-03-11 09:23:22

Go語言面向?qū)ο?/a>

2021-06-02 09:23:57

Go開發(fā)內(nèi)存

2024-05-14 08:47:01

JavaPAC++

2025-01-26 10:35:42

Go語言類型

2023-03-06 07:50:19

內(nèi)存回收Go

2021-04-25 08:58:00

Go拍照云盤

2013-07-30 09:27:32

Go云計(jì)算語言

2013-07-30 09:23:43

VMwareGoogle云平臺

2011-12-06 11:12:59

Java

2022-09-02 08:17:40

MapStruct代碼工具

2024-10-28 11:39:30

2024-10-29 10:30:57

2011-10-08 14:09:27

JavaScript

2009-07-06 15:47:11

Servlet API

2024-01-08 07:02:48

數(shù)據(jù)設(shè)計(jì)模式

2024-11-11 10:46:05

2018-07-27 10:39:13

對象存儲Git
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號