Go では、ポインターは変数を操作するために不可欠です。ただし、異なるポインター型間のニュアンスを理解するのは難しい場合があります。この記事では、ポインターの違いと、fmt パッケージのデフォルトの書式設定を使用する場合の出力への影響を示す特定のシナリオについて説明します。
次のコード スニペットは、 issue:
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 の出力はアドレスの 16 進表現です。
fmt.Printf 関数は %v を使用します。デフォルトの書式設定の動詞。出力される値のタイプに基づいて特定の書式を選択します。ポインターの場合、デフォルトの形式はアドレスの 16 進表現です。
最初のケース (ポインターとしての test1) では、出力される値は Test 構造体へのポインターです。ただし、構造体はゼロ値で初期化されているため、出力は空です。
2 番目のケース (ポインタとしての test2) では、出力される値はインターフェイスへのポインタです。{} %v 動詞に到達する前に、値は別のインターフェース内で追加のラッピングを通過し、Models["test"] を指します。{}最終的に出力される値はインターフェイスへのポインタであるため、ポインタのデフォルトの書式設定が適用され、アドレスの 16 進数表現が得られます。
この問題に対処するには、次のようにします。型アサーションを使用して、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{} でラップすることで、アドレスの 16 進表現を回避します。
以上がGo の `fmt.Printf` とポインター型はどのように相互作用するのでしょうか?また、異なるポインター型に対して異なる出力が生成されるのはなぜですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。