为了账号安全,请及时绑定邮箱和手机立即绑定

我如何将工人返回到 Go 中的工人池

我如何将工人返回到 Go 中的工人池

Go
千巷猫影 2023-06-12 17:13:33
我正在实施一个可以从渠道获取工作的工作人员池。一直超时后,我发现当一个worker fcn内部发生panic时,即使我做了恢复机制,worker还是不会再回到pool中。在 golang 操场上,我能够重现这个问题:修改后的游乐场代码:package mainimport "fmt"import "time"import "log"func recovery(id int, results chan<- int) {    if r := recover(); r != nil {        log.Print("IN RECOVERY FUNC - Failed worker: ",id)          results <- 0    }}func worker(id int, jobs <-chan int, results chan<- int) {    for j := range jobs {    defer recovery(id, results)    if id == 1 {        panic("TEST")    }        fmt.Println("worker", id, "started job", j)        time.Sleep(time.Second)        fmt.Println("worker", id, "finished job", j)        results <- j * 2    }}func main() {    jobs := make(chan int, 100)    results := make(chan int, 100)    for w := 1; w <= 3; w++ {        go worker(w, jobs, results)    }    for j := 1; j <= 10; j++ {        jobs <- j    }    close(jobs)    for a := 1; a <= 10; a++ {        <-results    }}为了测试,我在使用 worker 1 时实施了恐慌。运行时,函数会按预期发生恐慌,并按预期进入恢复状态(也不会将值推送到通道中),但是 worker 1 似乎永远不会回来。没有恐慌的输出:worker 3 started job 1worker 1 started job 2worker 2 started job 3worker 1 finished job 2worker 1 started job 4worker 3 finished job 1worker 3 started job 5worker 2 finished job 3worker 2 started job 6worker 3 finished job 5worker 3 started job 7worker 1 finished job 4worker 1 started job 8worker 2 finished job 6worker 2 started job 9worker 1 finished job 8worker 1 started job 10worker 3 finished job 7worker 2 finished job 9worker 1 finished job 10恐慌输出:worker 3 started job 12009/11/10 23:00:00 RECOVERY Failed worker: 1worker 2 started job 3worker 2 finished job 3worker 2 started job 4worker 3 finished job 1worker 3 started job 5worker 3 finished job 5worker 3 started job 6worker 2 finished job 4worker 2 started job 7worker 2 finished job 7worker 2 started job 8worker 3 finished job 6worker 3 started job 9worker 3 finished job 9worker 3 started job 10worker 2 finished job 8worker 3 finished job 10恢复后(或在恢复过程中)如何将工作人员 1 返回池中
查看完整描述

1 回答

?
翻翻过去那场雪

TA贡献2065条经验 获得超13个赞

如果你关心这些错误,你可以将一个errors通道传递给辅助函数,如果他们遇到一个error,将它发送到通道然后continue。主循环可以处理这些错误。

或者,如果您不关心错误,只需continue跳过该作业即可。

continue语句基本上停止处理循环的迭代,并继续下一个。


查看完整回答
反对 回复 2023-06-12
  • 1 回答
  • 0 关注
  • 115 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信