Dalam Go, pernyataan "tunda" membolehkan kami menjadualkan fungsi untuk dilaksanakan sejurus sebelum fungsi sekeliling kembali. Walau bagaimanapun, ini boleh menyebabkan kekeliruan apabila berurusan dengan pembolehubah yang diubah suai dalam fungsi lampiran.
Pertimbangkan fungsi berikut:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
Mengikut Spesifikasi Go, apabila pernyataan "menangguhkan" dilaksanakan, nilai fungsi dan parameter dinilai dan disimpan untuk kemudian perlaksanaan. Ini bermakna apabila fungsi akhirnya dipanggil, nilai x akan tetap 0 kerana ia dinilai pada masa penangguhan.
Untuk menyelesaikan isu ini, kami boleh menggunakan fungsi tanpa nama dalam pernyataan "tunda":
defer func() { fmt.Println(x) }()
Di sini, x bukan parameter fungsi tanpa nama, jadi ia tidak akan dinilai apabila pernyataan "menangguhkan" dilaksanakan. Sebaliknya, nilai x akan ditangkap pada masa fungsi tanpa nama dipanggil, memastikan nilai yang paling terkini dicetak.
Menggunakan Penunjuk:
var x int defer func() { fmt.Println(&x) }()
Pendekatan ini menggunakan penunjuk kepada x sebagai parameter fungsi tertunda. Apabila pernyataan "menangguhkan" dilaksanakan, hanya penunjuk yang dinilai, bukan nilai x. Apabila fungsi tertunda dipanggil, ia akan mengakses nilai semasa x melalui penuding.
Menggunakan Jenis Tersuai:
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } var x MyInt defer fmt.Println(&x)
Penyelesaian ini serupa dengan pendekatan penunjuk tetapi menggunakan jenis tersuai (MyInt) yang melaksanakan kaedah String(). Dengan melaksanakan String(), kita boleh mengawal cara nilai x dicetak.
Menggunakan Slice:
var x []int defer fmt.Println(x)
Menghiris ialah jenis deskriptor dalam Go, yang bermaksud bahawa nilainya ialah rujukan kepada tatasusunan asas. Apabila kita menangguhkan kepingan, hanya rujukan yang dinilai, bukan elemen tatasusunan sebenar. Akibatnya, sebarang perubahan yang dibuat pada elemen hirisan selepas penangguhan akan ditunjukkan dalam output bercetak.
Membungkus dalam Struktur:
type Wrapper struct { Value int } var x Wrapper defer fmt.Println(&x)
Pendekatan ini serupa dengan menggunakan penuding, tetapi kami membungkus nilai dalam struct untuk mengelakkan daripada menyahrujuk penuding dalam penundaan fungsi.
Atas ialah kandungan terperinci Bagaimana untuk Menggunakan `defer` dengan betul dalam Go dengan Pembolehubah Diubah Suai?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!