簡單介紹Golang列印複雜結構體的兩種方法
導讀 | 在 Golang 語言開發中,我們經常會使用結構體型別,如果我們使用的結構體型別的變數包含指標型別的欄位,我們在記錄日誌的時候,指標型別的欄位的值是指標地址,將會給我們 debug 程式碼造成不便 |
在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列印複雜結構體兩種方法詳解的文章就介紹到這了
原文來自:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2937989/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 簡單介紹C語言使用四種方法初始化結構體C語言結構體
- 簡單介紹Golang切片刪除指定元素的三種方法Golang
- 簡單介紹MySQL列印死鎖日誌的方法MySql
- 簡單介紹如何使用Bazel構建Golang程式Golang
- 簡單介紹MySQL開啟事務的兩種方式MySql
- 軟體開發常用結構以及SSM框架的簡單介紹SSM框架
- 簡單介紹python連線telnet和ssh的兩種方式Python
- NodeJS專案基礎結構簡單介紹NodeJS
- 簡單介紹四種Python 列表反轉顯示的方法Python
- 簡單介紹NMS的實現方法
- 簡單介紹PostgreSQL解析URL的方法SQL
- 簡單介紹oracle重置序列的方法Oracle
- 簡單介紹java中的equals()方法Java
- Vue3專案的簡單搭建與專案結構的簡單介紹Vue
- 簡單介紹redux的中介軟體Redux
- CSRF簡單介紹及利用方法
- 工控機的軟體的簡單介紹
- javascript演算法的複雜度介紹JavaScript演算法複雜度
- SVG程式碼構成簡單介紹SVG
- 簡單介紹架構設計的原則!架構
- 簡單介紹MySQL索引失效的幾種情況MySql索引
- Webpack 的簡單介紹Web
- AOP的簡單介紹
- Django 構建模板form表單的兩種方法DjangoORM
- form表單的簡單介紹ORM
- Flownet 介紹 及光流的簡單介紹
- 簡單介紹常見的三種架構設計模式及其優缺點!架構設計模式
- golang toolkits包的使用方法介紹Golang
- 簡單介紹2種Java讀取Oracle大欄位資料(CLOB)的方法JavaOracle
- 簡單介紹下各種 JavaScript 解析器JavaScript
- 簡單介紹redis加鎖常用幾種方式Redis
- JavaScript中的幾種資料結構簡介JavaScript資料結構
- vue匯出excel(簡單方法完整介紹)VueExcel
- 簡單介紹js 陣列 fill() 填充方法JS陣列
- 四種許可權修飾符的簡單介紹
- Maven簡單介紹——必要的Java管理與構建MavenJava
- 簡單介紹python中的單向連結串列實現Python
- RPC簡單介紹RPC