Rumah >pembangunan bahagian belakang >Golang >io.Paip dan kebuntuan apabila cuba menulis/membaca

io.Paip dan kebuntuan apabila cuba menulis/membaca

王林
王林ke hadapan
2024-02-06 09:40:031172semak imbas

尝试写入/读取时 io.Pipe 和死锁

Kandungan soalan

Saya menghabiskan masa berjam-jam cuba memahami logik asas tetapi tidak membuat kemajuan. Kod di bawah mengembalikan kebuntuan selepas lelaran pertama. Jika saya tutup penulis sebelum io.copy kebuntuan hilang tetapi tiada apa yang dicetak (kerana hujung tulisan paip ditutup sebelum membaca)

func main() {
    reader, writer := io.pipe()
    c := make(chan string)

    go func() {
        for i := 0; i < 5; i++ {
            text := fmt.sprintf("hello %vth time", i+1)
            c <- text
        }

        close(c)
    }()

    for msg := range c {
        msg = fmt.sprintf("\nreceived from channel -> %v\n", msg)

        go fmt.fprint(writer, msg)
        io.copy(os.stdout, reader)
        writer.close()
    }

}

Ini ialah ralat selepas menjalankan kod

received from channel -> hello 1th time fatal error: all goroutines
are asleep - deadlock!
goroutine 1 [select]: io.(*pipe).read(0xc000130120, {0xc00013e000,
0x8000, 0xc00011e001?})
/usr/lib/go/src/io/pipe.go:57 +0xb1 io.(*PipeReader).Read(0x0?, {0xc00013e000?, 0xc00011e050?, 0x10?})
/usr/lib/go/src/io/pipe.go:136 +0x25 io.copyBuffer({0x4bde98, 0xc00011e050}, {0x4bddb8, 0xc00012e018}, {0x0, 0x0, 0x0})
/usr/lib/go/src/io/io.go:427 +0x1b2 io.Copy(...)
/usr/lib/go/src/io/io.go:386 os.genericReadFrom(0x101c00002c500?, {0x4bddb8, 0xc00012e018})
/usr/lib/go/src/os/file.go:161 +0x67 os.(*File).ReadFrom(0xc00012e008, {0x4bddb8, 0xc00012e018})
/usr/lib/go/src/os/file.go:155 +0x1b0 io.copyBuffer({0x4bde38, 0xc00012e008}, {0x4bddb8, 0xc00012e018}, {0x0, 0x0, 0x0})
/usr/lib/go/src/io/io.go:413 +0x14b io.Copy(...)
/usr/lib/go/src/io/io.go:386 main.pipetest()
/home/stranger/source-code/golang/ipctest/pipes/main.go:39 +0x1ae main.main()
/home/stranger/source-code/golang/ipctest/pipes/main.go:10 +0x17
goroutine 18 [chan send]: main.pipetest.func1()
/home/stranger/source-code/golang/ipctest/pipes/main.go:29 +0x85 created by main.pipetest
/home/stranger/source-code/golang/ipctest/pipes/main.go:26 +0x17a exit status 2

Jawapan betul


io.copy 不断尝试复制,直到读取器到达 eof(在本例中,当管道关闭时)。由于您在 io.copy 结束后调用 writer.close() after ,因此 io.copy Teruskan cuba menyalin sehingga pembaca mencapai eof (dalam kes ini, apabila paip ditutup). Memandangkan anda memanggil writer.close() selepas selepas

selesai,

tidak akan pernah melihat eof dan hang selama-lamanya. closeable 对象只应关闭一次,并且在 close

Satu lagi masalah dengan kod anda ialah anda cuba menutup paip beberapa kali (setiap kali kod gelung berulang). Secara amnya, ketidaktersediaan diandaikan selepas

d. Jika anda perlu menggunakannya semula, anda harus membuat contoh baharu.

🎜Ini ialah semakan kod yang berfungsi: 🎜
func main() {
    c := make(chan string)

    go func() {
        for i := 0; i < 5; i++ {
            text := fmt.Sprintf("hello %vth time", i+1)
            c <- text
        }

        close(c)
    }()

    for msg := range c {
        msg = fmt.Sprintf("\nreceived from channel -> %v\n", msg)

        // Create a new pipe for this message.
        reader, writer := io.Pipe()
        go func() {
            fmt.Fprint(writer, msg)
            // Close the pipe after writing the message.
            writer.Close()
        }()

        io.Copy(os.Stdout, reader)
    }
}

Atas ialah kandungan terperinci io.Paip dan kebuntuan apabila cuba menulis/membaca. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan:
Artikel ini dikembalikan pada:stackoverflow.com. Jika ada pelanggaran, sila hubungi admin@php.cn Padam