Pembungkusan Ralat Rekursif dalam Go
Mekanisme pengendalian ralat Go menyediakan kaedah seperti Is() dan As() untuk pemeriksaan ralat. Walau bagaimanapun, sering diandaikan bahawa kaedah ini menyokong pembalut ralat rekursif. Ini tidak sepenuhnya benar.
Menggunakan fmt.Errorf dengan kata kerja %w hanya membenarkan pembalut ralat cetek. Untuk mencapai rekursi sebenar, jenis ralat tersuai diperlukan.
Lazimnya, jenis ralat tersuai akan menyertakan penunjuk kepada ralat yang dibalut serta kaedah untuk membukanya. Walau bagaimanapun, hanya membungkus ralat dengan cara ini menimbulkan masalah. Jika berbilang ralat yang dibalut berkongsi penunjuk yang sama, perbandingan ralat akan gagal kerana ralat Go dibandingkan mengikut alamat.
Penyelesaian adalah dengan melaksanakan kaedah Is() dan As() dalam jenis ralat tersuai. Kaedah ini harus membandingkan ralat terbenam dan bukannya jenis tersuai itu sendiri, membenarkan pembalut ralat rekursif penuh tanpa isu berasaskan alamat.
Sebagai contoh, berikut ialah versi ringkas jenis ralat yang menyokong pembalut rekursif:
type errorChain struct { err error next *errorChain } func (c errorChain) Is(err error) bool { return errors.Is(c.err, err) } func (c errorChain) As(target any) bool { return errors.As(c.err, target) } func (c errorChain) Unwrap() error { return c.next }
Jenis ralat ini kemudiannya boleh digunakan untuk membalut ralat seperti yang dikehendaki, membenarkan pemeriksaan ralat penuh menggunakan Is() dan As().
Atas ialah kandungan terperinci Adakah `errors.Is` dan `errors.As` Go Menyokong Pembalut Ralat Rekursif?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!