Go 言語で defer
キーワードを使用すると、関数の終了までコードの実行が遅れる可能性があります。開発では、開いているファイル記述子を閉じる、接続を閉じる、リソースを解放するなど、その後の作業を完了するために defer
キーワードを使用することがよくあります。
func demo0() { fileName := "./test.txt" f, _ := os.OpenFile(fileName, os.O_RDONLY, 0) defer f.Close() contents, _ := ioutil.ReadAll(f) fmt.Println(string(contents))}
defer
キーワードは通常、リソースを解放し忘れることを防ぐために、リソースを開くコードの直後に続きます。defer で宣言されたコードは、関数が終了するまで実際には実行されません。 defer はシンプルで使いやすいですが、 しかし、その機能を無視すると、開発中に混乱に直面することになります。そこで、defer の 5 つの主要な機能をまとめ、8 つのデモを通して defer の機能を段階的に紹介しました。
first-in-last-out 機能と同様に、この defer の機能も理解しやすいです。 最初に によって開かれたリソースは、後続のコードによって依存される可能性があるため、 の後に を放しても安全です。
func demo1() { for i := 0; i < 5; i++ { defer fmt.Println("defer:", i) }}// defer: 4// defer: 3// defer: 2// defer: 1// defer: 0機能 2: スコープは現在の関数であり、さまざまな関数の下にさまざまな遅延スタックがありますデモ 2 を実行します。結果から、最初の匿名関数と 2 番目の匿名関数が実行されていることがわかります。匿名関数 関数の遅延実行の順序は重要ではありません。
遅延スコープは現在の関数のみであり、現在の関数の最後に実行されるため、関数ごとに異なる遅延スタックが存在します。
func demo2() { func() { defer fmt.Println(1) defer fmt.Println(2) }() fmt.Println("=== 新生代农民工啊 ===") func() { defer fmt.Println("a") defer fmt.Println("b") }()}// 2// 1// === 新生代农民工啊 ===// b// a
仮パラメータnの値は、実行時のではなく、宣言時に確定するため、後続の変数numがどのように変化しても出力結果には影響しません。延期の。
func demo3_1() { num := 0 defer func(n int) { fmt.Println("defer:", n) }(num) // 等同 defer fmt.Println("defer:", num) for i := 0; i < 10; i++ { num++ } fmt.Println(num)}//10//defer: 0demo3_2 を実行します。ここでの defer の最終出力結果が変数 num と同じなのはなぜですか?ここではポインタが使用されているためです。
defer
を宣言すると、仮引数 p ポインタが指すアドレスが変数 num を指していることが確認され、変数 num が変化します。したがって、defer が実行されると、出力は p ポインタが指す変数 num の現在の値になります。 func demo3_2() {
num := 0
p := &num defer func(p *int) {
fmt.Println("defer:", *p)
}(p)
for i := 0; i < 10; i++ {
num++
}
fmt.Println(*p)}//10//defer: 10
defer3_3 をもう一度見てください。defer によって出力された変数は、関数パラメータを介して渡されません。「グローバル変数」num は defer
func demo3_3() { num := 0 defer func() { fmt.Println("defer:", num) }() for i := 0; i < 10; i++ { num++ } fmt.Println(num)}//10//defer: 10機能 4: return と defer の実行順序: return first defer thendefer と return は関数の最後に実行されますが、return が前に実行されることがわかります。 defer;
func demo4_1() (int, error) { defer fmt.Println("defer") return fmt.Println("return")}// return// defer
からも明らかですが、returnとdeferの実行順序と
**関数の戻り値** 「会う」、多くの複雑なシナリオが発生します。 demo4_2 では、関数は を使用して戻り値 に名前を付け、最終的な出力結果は 7 です。
(その後) return 後に defer が実行され、変数 num が取得されて変更され、値は 7;
func demo4_2() (num int) { num = 10 defer func() { num += 5 }() return 2}// 7
を使用し、最終的な結果出力は 2 です。プロセスは次のとおりです:
func demo4_3() int { num := 10 defer func() { num += 5 }() return 2}// 2
func demo5_1() { defer fmt.Println(1) defer fmt.Println(2) defer fmt.Println(3) panic("没点赞异常") // 触发defer出栈执行 defer fmt.Println(4) // 得不到执行}
func demo5_2() { defer func() { if err := recover(); err != nil { fmt.Println(err, "问题不大") } }() panic("没点赞异常") // 触发defer出栈执行 // ...}添付完全なコード:
以上が8 つのデモを使用して Go 言語の defer の 5 つの主要な機能を理解するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。