首頁 > 後端開發 > Golang > Go 的「fmt.Printf」和指標類型如何交互,以及為什麼它們會針對不同的指標類型產生不同的輸出?

Go 的「fmt.Printf」和指標類型如何交互,以及為什麼它們會針對不同的指標類型產生不同的輸出?

Linda Hamilton
發布: 2024-12-09 20:21:18
原創
596 人瀏覽過

How Do Go's `fmt.Printf` and Pointer Types Interact, and Why Do They Produce Different Outputs for Different Pointer Types?

評估 Go 指標的差異

在 Go 中,指針對於處理變數至關重要。然而,理解不同指針類型之間的細微差別可能具有挑戰性。本文探討了一個特定的場景,展示了指標之間的差異以及在使用 fmt 套件中的預設格式時它們如何影響輸出。

問題中的程式碼

以下程式碼片段示範了問題:

type Test struct {
    Test string
}

var Models = map[string]interface{}{
    "test": newTest(),
}

func main() {
    test1 := Test{}
    fmt.Println("Test 1: ")
    fmt.Printf("%v", test1)
    fmt.Println()
    fmt.Println("Test 1 as pointer: ")
    fmt.Printf("%v", &test1)
    fmt.Println()
    test2 := Models["test"]
    fmt.Println("Test 2: ")
    fmt.Printf("%v", test2)
    fmt.Println()
    fmt.Println("Test 2 as pointer: ")
    fmt.Printf("%v", &test2)
}

func newTest() Test {
    var model Test
    return model
}
登入後複製

問題

執行程式碼時,您會注意到輸出有所不同將test2 與test1 作為指標列印。 test1 作為指標的輸出是空字串,而 test2 作為指標的輸出是位址的十六進位表示法。

說明

fmt.Printf 函數使用 %v預設格式的動詞,它根據要列印的值的類型選擇特定格式。對於指針,預設格式是位址的十六進位表示形式。

在第一種情況(test1 作為指標)中,列印的值是指向 Test 結構的指標。但是,由於該結構體是用零值初始化的,因此輸出為空。

在第二種情況(test2 作為指標)中,列印的值是指向 interface{} 的指標。在到達 %v 動詞之前,該值在另一個 interface{} 內進行了額外的包裝,然後指向 Models["test"]。由於最終列印的值是指向 interface{} 的指針,因此將套用指針的預設格式,並且您將獲得位址的十六進位表示形式。

要解決此問題,需要使用類型斷言從 test2 中提取實際的 Test 結構。這可以透過以下方式實現:

t2 := Models["test"]
test2 := t2.(Test) // test2 is now of type Test
登入後複製

透過類型斷言,test2 變數現在指向 Test 結構,您可以像 test1 一樣列印它。

或者,可以使用透過在映射中儲存*Test 值來直接指向Test 的指標:

var Models = map[string]*Test{
    "test": newTest(),
}
登入後複製

這種方法消除了類型斷言和包裝的需要在interface{}中,從而避免了位址的十六進位表示。

以上是Go 的「fmt.Printf」和指標類型如何交互,以及為什麼它們會針對不同的指標類型產生不同的輸出?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板