Golang打印復(fù)雜結(jié)構(gòu)體兩種方法詳解
fmt結(jié)構(gòu)體占位符
在Golang中有原生的 fmt 格式化工具去打印結(jié)構(gòu)體,可以通過占位符%v、%+v、%#v去實(shí)現(xiàn),這3種的區(qū)別如下所示:
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) }
打印結(jié)果如下所示:
{張三 95}
{Name:張三 Age:95}
main.User{Name:"張三", Age:95}
其中的區(qū)別:
- %v占位符是不會(huì)打印結(jié)構(gòu)體字段名稱的,字段之間以空格隔開;
- %+v占位符會(huì)打印字段名稱,字段之間也是以空格隔開;
- %#v占位符則會(huì)打印結(jié)構(gòu)體類型和字段名稱,字段之間以逗號(hào)分隔
打印復(fù)雜結(jié)構(gòu)體
當(dāng)結(jié)構(gòu)體中的字段是指針類型時(shí),用占位符直接打印出來的是怎樣的呢?
還是以前面的例子為基礎(chǔ),我們給“張三”加一條狗,其中 User 結(jié)構(gòu)體中引入的 Dog 是指針類型,代碼如下:
type Dog struct { Name string Age int } type User struct { Name string Age int Dog *Dog } func main() { dog := Dog{ Name: "旺財(cái)", 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) }
這時(shí)還能把所有值打印出來嗎?
{張三 95 0xc000004078}
{Name:張三 Age:95 Dog:0xc000004078}
main.User{Name:"張三", Age:95, Dog:(*main.Dog)(0xc000004078)}
這時(shí)可以看到Dog字段打印的不是Dog結(jié)構(gòu)體內(nèi)部的值,而是一個(gè)地址值。很顯然,這個(gè)不是我們需要在日志中看到的,我們需要看的是結(jié)構(gòu)體具體的值,那這個(gè)值又怎么打印呢?
方案一
實(shí)現(xiàn) String() 或GoString() 方法
Golang 中的 fmt 包中有一個(gè) Stringer 接口,接口中只有一個(gè) 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 結(jié)構(gòu)體分別實(shí)現(xiàn) String() 方法,這種方法類似于 Java 中的 toString() 方法?;谇懊娴拇a,我們?cè)黾尤缦?String() 方法實(shí)現(xiàn):
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() + "}" }
運(yùn)行后,打印的結(jié)果如下所示:
{張三 95 {"name": "旺財(cái)","age": "2"}}
{Name:張三 Age:95 Dog:{"name": "旺財(cái)","age": "2"}}
main.User{Name:"張三", Age:95, Dog:(*main.Dog)(0xc000004078)}
發(fā)現(xiàn),實(shí)現(xiàn) String() 方法只對(duì) %v 和 %+v 占位符有效,對(duì)于%#v 占位符,其打印的結(jié)構(gòu)體指針類型還是一個(gè)地址值。
其實(shí)在 fmt 包中,Stringer 接口 下面,我們還可以看到另外一個(gè) 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 方法用于打印作為操作數(shù)傳遞給 %#v 格式的值)
找到了,我們?cè)賹?shí)現(xiàn) GoString() 方法,就可以用 %#v 占位符打印結(jié)構(gòu)體指針類型中的值了。
基于之前代碼增加如下代碼:
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() + "}" }
運(yùn)行后,打印結(jié)果如下所示,這下子就都可以打印了:
{張三 95 {"name": "旺財(cái)","age": "2"}}
{Name:張三 Age:95 Dog:{"name": "旺財(cái)","age": "2"}}
main.User{Name:"張三", Age:95, Dog:{"name": "旺財(cái)","age": "2"}}
到這里,我感覺這種方案有點(diǎn)麻煩呢,還有沒有其他不用維護(hù) String() 或 GoString() 的方法呢?
方案二
轉(zhuǎn)換成 json 格式
func main() { dog := Dog{ Name: "旺財(cái)", Age: 2, } user := User{ Name: "張三", Age: 95, Dog: &dog, } byteUser, _ := json.Marshal(&user) fmt.Println(string(byteUser)) }
打印結(jié)果如下所示,如果使用 json 庫的話,是可以直接把結(jié)構(gòu)體指針類型的具體值都打印出來的,比較方便:
{"Name":"張三","Age":95,"Dog":{"Name":"旺財(cái)","Age":2}}
到此這篇關(guān)于Golang打印復(fù)雜結(jié)構(gòu)體兩種方法詳解的文章就介紹到這了,更多相關(guān)Go打印復(fù)雜結(jié)構(gòu)體內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go語言排序算法之插入排序與生成隨機(jī)數(shù)詳解
從這篇文章開始將帶領(lǐng)大家學(xué)習(xí)Go語言的經(jīng)典排序算法,比如插入排序、選擇排序、冒泡排序、希爾排序、歸并排序、堆排序和快排,二分搜索,外部排序和MapReduce等,本文將先詳細(xì)介紹插入排序,并給大家分享了go語言生成隨機(jī)數(shù)的方法,下面來一起看看吧。2017-11-11解決golang 反射interface{}做零值判斷的一個(gè)重大坑
這篇文章主要介紹了解決golang 反射interface{}做零值判斷的一個(gè)重大坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-04-04B站新一代 golang規(guī)則引擎gengine基礎(chǔ)語法
這篇文章主要為大家介紹了B站新一代 golang規(guī)則引擎gengine基礎(chǔ)語法,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12golang替換無法顯示的特殊字符(\u0000,?\000,?^@)
這篇文章主要介紹了golang替換無法顯示的特殊字符,包括的字符有\(zhòng)u0000,?\000,?^@等,下文詳細(xì)資料,需要的小伙伴可以參考一下2022-04-04Golang實(shí)現(xiàn)常見排序算法的示例代碼
現(xiàn)在的面試真的是越來越卷了,算法已經(jīng)成為了面試過程中必不可少的一個(gè)環(huán)節(jié),你如果想進(jìn)稍微好一點(diǎn)的公司,算法是必不可少的一個(gè)環(huán)節(jié)。本文為大家準(zhǔn)備了Golang實(shí)現(xiàn)常見排序算法的示例代碼,需要的可以參考一下2022-05-05golang連接MongoDB數(shù)據(jù)庫及數(shù)據(jù)庫操作指南
MongoDB是Nosql中常用的一種數(shù)據(jù)庫,下面這篇文章主要給大家介紹了關(guān)于golang連接MongoDB數(shù)據(jù)庫及數(shù)據(jù)庫操作的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-09-09