go - 即使关闭, channel 也不会消亡

标签 go goroutine channels

在下面的代码中,我试图写入所有发送到 inputs channel 的文件,并通过 operationOutcomes channel 发送相应的响应

ma​​in.go

package main

import(
    lr "github.com/fabulousduck/librarian"
    "fmt"
)

func main() {
    writeOpCount := 100;
    operationOutcomes, inputs := make(chan lr.WriteOpResponse), make(chan lr.WriteOp)
    go lr.WriteC(inputs, operationOutcomes)

    for i := 0; i < writeOpCount; i++ {
        inputs <- lr.WriteOp{ Dest: `../exampleFiles/createdFiles/{i}.txt`, Content: `Invoice #{i}` }
    }

    close(inputs)  

    for i := 0; i < writeOpCount; i++ {
        writeResult := <-operationOutcomes
        fmt.Println("Response from write operation : ", writeResult.Msg, "err ", writeResult.Err, "bytes written : ", writeResult.BytesWritten)
    }

    close(operationOutcomes)
}

librarian.go 包库管理员

import(
    "os"
    "fmt"
)

type WriteOp struct {
    Dest, Content string
}

type WriteOpResponse struct {
    Msg error
    Err bool
    BytesWritten int
}

func WriteC (inputChannel <-chan WriteOp, outputChannel chan<- WriteOpResponse) {
    workOp :=  <-inputChannel
    go writeWorker(workOp, outputChannel)
}

func writeWorker (job WriteOp, outGoing chan<- WriteOpResponse) {
    file, err := os.OpenFile(job.Dest, os.O_RDWR, 0666)
    if err != nil {
        fmt.Println("err : ", err)
        outGoing <- WriteOpResponse{ Msg: err, Err: true, BytesWritten: 0 }
    }
    bytesWritten , err := file.WriteString(job.Content)
    if err != nil {
        outGoing <- WriteOpResponse{ Msg: err, Err: true, BytesWritten: 0 }
    }
    outGoing <- WriteOpResponse{ Msg: nil, Err: false, BytesWritten: bytesWritten } 
}

这在两种情况下都会引发死锁,无论有没有错误,即使我在完成处理后关闭了两个 channel ?

最佳答案

您最多只能从 operationOutcomes 中读取 writeOpCount 次,即使在 writeWorker 中每次执行最多可以写入 3 条消息该 channel (您的错误情况都不会导致函数返回,它会继续处理)。因为它是无缓冲的并且它停止被读取,所以在某些时候写入它的工作人员不能再添加消息并因此永远锁定。

此外,由于您只调用一次 WriteC 并且它不会循环,因此它只会读取和处理来自 inputs 的一条消息。如果 writeOpCount > 1,那么当它尝试对第二条消息进行排队时,它将永远锁定在第一个循环中。

关于go - 即使关闭, channel 也不会消亡,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44368901/

相关文章:

unit-testing - 不运行名称为 package_test.go 的程序

string - 如何获取字符串中的行数?

Golang 在 goroutine 之间共享大量数据

go - 总是与 channel 陷入僵局

go - beego中如何指定自定义迁移表

haskell - 如何使用 Haskell 模拟 Go 的 channel ?

select - 当涉及多个 channel 时,select 如何工作?

go - 理解 WaitGroups 的逻辑

c++ - opencv矩阵多 channel 获取和填充

go - 如何将 Base64 编码的 p12 转换为 tls.Certificate