gpt4 book ai didi

go - 当 channel 关闭时,以接收 channel 作为参数的 goroutines 是否停止?

转载 作者:数据小太阳 更新时间:2023-10-29 03:40:40 25 4
gpt4 key购买 nike

一直在看《用go构建微服务》,书中介绍了 apache/go-resiliency/deadline 用于处理超时的包。

deadline.go

// Package deadline implements the deadline (also known as "timeout") resiliency pattern for Go.
package deadline

import (
"errors"
"time"
)

// ErrTimedOut is the error returned from Run when the deadline expires.
var ErrTimedOut = errors.New("timed out waiting for function to finish")

// Deadline implements the deadline/timeout resiliency pattern.
type Deadline struct {
timeout time.Duration
}

// New constructs a new Deadline with the given timeout.
func New(timeout time.Duration) *Deadline {
return &Deadline{
timeout: timeout,
}
}

// Run runs the given function, passing it a stopper channel. If the deadline passes before
// the function finishes executing, Run returns ErrTimeOut to the caller and closes the stopper
// channel so that the work function can attempt to exit gracefully. It does not (and cannot)
// simply kill the running function, so if it doesn't respect the stopper channel then it may
// keep running after the deadline passes. If the function finishes before the deadline, then
// the return value of the function is returned from Run.
func (d *Deadline) Run(work func(<-chan struct{}) error) error {
result := make(chan error)
stopper := make(chan struct{})

go func() {
result <- work(stopper)
}()

select {
case ret := <-result:
return ret
case <-time.After(d.timeout):
close(stopper)
return ErrTimedOut
}
}

deadline_test.go

package deadline

import (
"errors"
"testing"
"time"
)

func takesFiveMillis(stopper <-chan struct{}) error {
time.Sleep(5 * time.Millisecond)
return nil
}

func takesTwentyMillis(stopper <-chan struct{}) error {
time.Sleep(20 * time.Millisecond)
return nil
}

func returnsError(stopper <-chan struct{}) error {
return errors.New("foo")
}

func TestDeadline(t *testing.T) {
dl := New(10 * time.Millisecond)

if err := dl.Run(takesFiveMillis); err != nil {
t.Error(err)
}

if err := dl.Run(takesTwentyMillis); err != ErrTimedOut {
t.Error(err)
}

if err := dl.Run(returnsError); err.Error() != "foo" {
t.Error(err)
}

done := make(chan struct{})
err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
return nil
})
if err != ErrTimedOut {
t.Error(err)
}
<-done
}

func ExampleDeadline() {
dl := New(1 * time.Second)

err := dl.Run(func(stopper <-chan struct{}) error {
// do something possibly slow
// check stopper function and give up if timed out
return nil
})

switch err {
case ErrTimedOut:
// execution took too long, oops
default:
// some other error
}
}

第一个问题

// in deadline_test.go
if err := dl.Run(takesTwentyMillis); err != ErrTimedOut {
t.Error(err)
}

我无法理解上述代码的执行流程。据我了解,因为 takesTwentyMillis函数休眠时间超过设置的超时持续时间 10 毫秒,

// in deadline.go
case <-time.After(d.timeout):
close(stopper)
return ErrTimedOut

time.After 发出当前时间,并选择这种情况。然后关闭 stopper channel 并返回 ErrTimeout。

我不明白的是,关闭停止 channel 对可能仍在运行的匿名 goroutine 有何影响我认为,当停止器 channel 关闭时,下面的 goroutine 可能仍在运行。

go func() {
result <- work(stopper)
}()

(这里说错了请指正)我想在close(stopper)之后,这个 goroutine 会调用 takesTwentyMillis (=功函数)以塞子 channel 作为其参数。该函数将继续并休眠 20 毫秒并返回 nil 以传递给结果 channel 。 main() 到此结束,对吧?

我不明白在这里关闭塞子 channel 有什么意义。 takesTwentyMillis函数似乎并没有在函数体内使用 channel :(。

第二题

// in deadline_test.go within TestDeadline()
done := make(chan struct{})
err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
return nil
})
if err != ErrTimedOut {
t.Error(err)
}
<-done

这是我不完全理解的部分。我想什么时候dl.Run运行时,停止 channel 被初始化。但是因为stopper channel中没有值,函数调用会阻塞在<-stopper处。 ...但是因为我不理解这段代码,所以我不明白为什么这段代码首先存在(即这段代码试图测试什么,以及它是如何执行的,等等)。


关于第二个问题的第三个(附加)问题

所以我明白当Run第二个问题中的函数触发停止器 channel 关闭,工作函数获取信号。然后 worker 关闭 done channel 并返回 nil。我使用 delve(=go debugger) 来查看,gdb 将我带到 deadline.go 中的 goroutine。在 return nil 行之后.

   err := dl.Run(func(stopper <-chan struct{}) error {
<-stopper
close(done)
--> return nil
})

在输入 n 跳到下一行后,delve 将我带到这里

    go func() {
--> result <- work(stopper)
}()

过程在这里结束了,因为当我再次输入 n 时,命令行提示 PASS 并且过程退出。为什么这个过程在这里结束? work(stopper)似乎返回 nil ,然后应该将其传递给结果 channel ,对吗?但是由于某种原因,这一行似乎没有执行。

我知道主 goroutine,它是 Run函数,已经返回 ErrTimedOut。所以我想这与此有关?

最佳答案

第一个问题

stopper的使用 channel 用于发出功能信号,例如takesTwentyMillis它的截止日期已经到了,调用者不再关心它的结果。通常这意味着 worker 函数像 takesTwentyMillis应该检查 stopper channel 已经关闭,因此它可能会取消它的工作。仍然,检查 stopper channel 是辅助功能的选择。它可能会或可能不会检查 channel 。

func takesTwentyMillis(stopper <-chan struct{}) error {
for i := 0; i < 20; i++ {
select {
case <-stopper:
// caller doesn't care anymore might as well stop working
return nil
case <-time.After(time.Second): // simulating work
}
}
// work is done
return nil
}

第二个问题

这部分Deadline.Run()将关闭塞子 channel 。

case <-time.After(d.timeout):
close(stopper)

在关闭的 channel ( <-stopper ) 上读取将立即返回该 channel 的零值。我认为它只是在测试最终超时的工作函数。

关于go - 当 channel 关闭时,以接收 channel 作为参数的 goroutines 是否停止?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53673690/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com