Rumah > pembangunan bahagian belakang > Golang > Mengapakah `json.NewDecoder().Decode()` tidak mengembalikan ralat tarikh akhir konteks apabila `ioutil.ReadAll()` melakukannya dalam tarikh akhir konteks dalam Go?

Mengapakah `json.NewDecoder().Decode()` tidak mengembalikan ralat tarikh akhir konteks apabila `ioutil.ReadAll()` melakukannya dalam tarikh akhir konteks dalam Go?

Barbara Streisand
Lepaskan: 2024-10-31 12:34:31
asal
527 orang telah melayarinya

Why does `json.NewDecoder().Decode()` not return a context deadline error when `ioutil.ReadAll()` does within a context deadline in Go?

Gelagat Tidak Dijangka json.NewDecoder().Decode() Dalam Tarikh Akhir Konteks

Program In Go dengan tarikh akhir konteks ditetapkan, respons badan yang dibaca oleh ioutil.ReadAll() dijangka akan mengembalikan ralat (context.DeadlineExceeded). Walau bagaimanapun, tingkah laku ini tidak dipatuhi apabila membaca badan tindak balas dengan json.NewDecoder(resp.Body).Decode(), yang mengembalikan sifar.

Mari kita mendalami isu ini:

Mengikut maklum balas, pakej net/http mungkin menggunakan penimbal untuk memproses permintaan. Ini memerlukan badan tindak balas yang masuk boleh dibaca sebahagian atau keseluruhannya dan ditimbal sebelum masa anda melakukannya. Oleh itu, konteks yang tamat tempoh mungkin tidak menghalang anda daripada melengkapkan bacaan kandungan.

Untuk menggambarkan perkara ini dengan lebih jelas, kami melaraskan contoh untuk melancarkan pelayan HTTP ujian yang sengaja melambatkan respons sebahagiannya:

<code class="go">ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
    s := []byte(`{&quot;ip&quot;:&quot;12.34.56.78&quot;}`)
    w.Write(s[:10])
    if f, ok := w.(http.Flusher); ok {
        f.Flush()
    }
    time.Sleep(time.Second * 6)
    w.Write(s[10:])
}))
defer ts.Close()
url = ts.URL

readDoesntFail()
readFails()</code>
Salin selepas log masuk

Pelayan ujian ini mengeluarkan objek JSON yang menyerupai respons ip.jsontest.com. Berbeza dengan itu, ia hanya menyampaikan 10 bait badan pada mulanya, kemudian mengepamnya dan tidur selama 6 saat dengan sengaja sebelum menghantar bakinya, memberikan masa pelanggan untuk tamat tempoh.

Apabila kami melaksanakan readDoesntFail() dengan pelayan ini , kami dapat:

before reading response body, context error is: context deadline exceeded
panic: Get &quot;http://127.0.0.1:38230&quot;: context deadline exceeded

goroutine 1 [running]:
main.readDoesntFail()
    /tmp/sandbox721114198/prog.go:46 +0x2b4
main.main()
    /tmp/sandbox721114198/prog.go:28 +0x93
Salin selepas log masuk

Sekarang, dengan contoh kami yang dikemas kini, json.Decoder.Decode() berusaha untuk membaca daripada sambungan kerana maklumat belum ditimbal, membenarkan tamat tempoh konteks menggesa ralat kerana konteks tamat tempoh.

Atas ialah kandungan terperinci Mengapakah `json.NewDecoder().Decode()` tidak mengembalikan ralat tarikh akhir konteks apabila `ioutil.ReadAll()` melakukannya dalam tarikh akhir konteks dalam Go?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan