
上下文取消后,<-ctx.done()未执行的原因
在 go 语言中,使用 context.context 包可以实现协程间通信和取消。当 context 被取消后,<-ctx.done() 会被唤醒并返回,但有时它可能不会被执行。
这是因为<-ctx.done()阻塞在读取ctx.done()通道上。如果在调用cancel()之前,ctx通道中的数据已经准备好,则<-ctx.done()将不会被唤醒。
导致该问题的一个示例
以下代码演示了该问题:
package main
import (
"context"
"fmt"
)
func gen(ctx context.context) <-chan interface{} {
ch := make(chan interface{})
go func() {
n := 0
for {
select {
case <-ctx.done():
fmt.println("done")
default:
n += 1
ch <- n
}
}
}()
return ch
}
func main() {
ctx, cancel := context.withcancel(context.background())
for n := range gen(ctx) {
fmt.println(n)
if n == 5 {
break
}
}
defer cancel()
}在这种情况下,for循环将在读取到第六个元素之前一直运行。这意味着,当cancel()被调用时,第六个元素已经准备好了。因此,<-ctx.done()不会被唤醒。
解决方法
可以通过关闭通道来解决此问题。当上下文被取消时,关闭通道将导致<-ctx.done()被唤醒。
以下代码显示了改进后的示例:
package main
import (
"context"
"fmt"
)
func gen(ctx context.Context) <-chan interface{} {
ch := make(chan interface{})
go func() {
n := 0
for {
select {
case <-ctx.Done():
fmt.Println("done")
close(ch) // 关闭通道,range结束
return
default:
n += 1
ch <- n
}
}
}()
return ch
}
func main() {
ctx, cancel := context.WithCancel(context.Background())
for n := range gen(ctx) {
fmt.Println(n)
if n == 5 {
cancel()
// break不能break,否则没有从ch中读取,就会一直阻塞在ch<-n
}
}
}这样,无论第六个元素是否已经准备好,<-ctx.done()都将被唤醒,并关闭通道。
以上就是Go 语言中,当 context 被取消后,`的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号