golang 中的协程使用非常方便,但是协程什么时候结束是一个控制问题,可以用 select 配合使用。
首先声明,golang 使用并不熟悉,本文仅仅是记录使用过程中遇到的一些坑。
子协程和父协程的通信通常用 context 或者 chan。我遇到一个通常的使用场景,在子协程中尝试多次处理,父协程等待一段时间超时,我选择用 chan 实现。我以为 select 和 C++ 中 switch 类似,所以最开始代码类似如下:文章来源地址https://www.yii666.com/blog/292779.html文章地址https://www.yii666.com/blog/292779.html
测试发现无法实现 timeout,又仔细查看文档,才发现 golang 中 select 另有玄机。废话少说,直接总结要点:
- select 中的 case 必须是进行 chan 的手法操作,也就是只能在 case 中操作 chan,并且是 非阻塞接收 。
- select 中的 case 是同时监听的,多个 case 同时操作,并未 switch 中一个个顺序判断。如果多个 case 满足要求,随机执行一个,如果一个没有则阻塞当前的协程(没有 default 情况下)。 很类似 Linux 文件符操作的 select 语义 。
- 上面说的阻塞是没有 default 的情况下,如果有 default,则执行 default,然后退出 select,也就是不会阻塞当前协程。
time.After
开一个新的协程去不断尝试,在外的三个 case 有一个满足,则会执行。但是这里有一个问题非常需要注意: 子协程什么时候退出? 。网址:yii666.com
case time.Afterdone
done := make(char int, 1)
还要一种办法,上面说了,select 操作 chan,并且可以指定 default,那是不是有思路了呢?
我们尝试往 chan 中发送,如果发不出去,则就退出,也实现了目的。文章来源地址:https://www.yii666.com/blog/292779.html
最后总结一下,goroutine 泄露的防范条例:网址:yii666.com<
- 创建 goroutine 时就要想好该 goroutine 该如何结束。
- 使用 chan 时,要考虑到 chan 阻塞时协程可能的行为。
- 实现循环语句时注意循环的退出条件,避免死循环。