在 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中文網其他相關文章!