文中例子参考:https://www.flysnow.org/2017/05/12/go-in-action-go-context.html
1. 使用 chan + select 控制 goroutine 停止
package main
import (
"fmt"
"time"
)
func main() {
stop := make(chan bool)
go func() {
for {
select {
case value1, ok1:= <-stop:
fmt.Println(value1, ok1)
fmt.Println("监控退出,停止了...1")
return
default:
fmt.Println("goroutine监控中...1")
time.Sleep(2 * time.Second)
}
}
}()
go func() {
for {
select {
case value2, ok2:= <-stop:
fmt.Println(value2, ok2)
fmt.Println("监控退出,停止了...2")
return
default:
fmt.Println("goroutine监控中...2")
time.Sleep(2 * time.Second)
}
}
}()
time.Sleep(10 * time.Second)
fmt.Println("可以了,通知监控停止")
close(stop)
//stop<- true
//为了检测监控过是否停止,如果没有监控输出,就表示停止了
time.Sleep(5 * time.Second)
}
运行这个示例代码我们会发现只有一个 goroutine 能正常停止,这里需要了解一点前置知识: golang 的 select 就是监听 IO 操作,当 IO 操作发生时,触发相应的动作
;当我们向通道 stop
传值,goroutine 内部的 select 操作会监控到这一行为,从而触发对应的操作:停止运行,退出;
那为什么只有一个 goroutine 能停止运行,另外一个没有收到停止讯号吗?是的! select 操作实际上还是从通道里读取数据,只是这个操作是非阻塞的,通道里有数据,那我就读取出来