GoLang 关闭(退出) goroutine 的方式_golang 外部退出go协程,2024年最新最新整理《Golang架构师面试题解析大全》

44 阅读1分钟
time.Sleep(3 \* time.Second)
// 向退出通道发送退出信号
ch <- true
fmt.Println("退出主函数!!!")

}


![在这里插入图片描述](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/3a564bbf6e974c07b8a59e985bd68eb6~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=ji982hFeHJDhSVciTpEqw8ytD6o%3D)


#### 2、关闭退出通道


  退出多个 goroutine。


  当 goroutine 过多时,我们就需要向退出通道中发送多次退出信号,如果再用发送信号的方式就很麻烦,有一个很简单的方法,关闭 channel,这样所有监听退出 channel 的 goroutine 就都会收到关闭信号。



func main() { // 定义退出通道 ch := make(chan bool) go func() { for { select { case <-ch: fmt.Println("收到退出信号01,退出协程01!!!") return default: fmt.Println("监控01!!!") time.Sleep(1 * time.Second) } } }() go func() { for { select { case <-ch: fmt.Println("收到退出信号02,退出协程02!!!") return default: fmt.Println("监控02!!!") time.Sleep(1 * time.Second) } } }() time.Sleep(3 * time.Second) // 关闭退出通道 close(ch) time.Sleep(5 * time.Second) fmt.Println("退出主函数!!!") }


![在这里插入图片描述](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/ea00ff4d390a44d1a111abc50af1cedd~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=7XyeMjYKLSES0xlwaSWAFtzT27Y%3D)


#### 3、使用context.WithCancel() 方法


  使用 context.WithCancel() 方法,手动调用 cancel() 方法退出 goroutine。


  只有创建它的函数才能调用取消函数来取消此 context。如果你愿意,可以传递取消函数,但是,强烈建议不要这样做,这可能导致取消函数的调用者没有意识到取消 context 的下游影响,可能存在源自此的其他 context,可能导致程序以意外的方式运行。简而言之,永远不要传递取消函数。



func main() { // 新建一个上下文 ctx := context.Background() // 在初始上下文的基础上创建一个有取消功能的上下文,需要取消时,就调用cancel(),发出取消信号 ctx, cancel := context.WithCancel(ctx) go func() { for { select { case <-ctx.Done(): fmt.Println("退出协程!!!") return default: fmt.Println("监控中 WithCancel !!!") time.Sleep(1 * time.Second) } } }()

time.Sleep(5 \* time.Second)
// 取消信号
cancel()
time.Sleep(2 \* time.Second)
fmt.Println("退出主函数!!!")

}


![在这里插入图片描述](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/0cb58b7f6ad14fc3bd67d1016a5cb7b0~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=rWrx7c7%2FST%2FWH5WSrEtiQpiZ2Mw%3D)


#### 4、使用context.WithTimeout() 方法


  使用context.WithTimeout() 方法,手动调用 cancel()方法,在超时之前退出 goroutine。



func main() { ctx, cancel := context.WithTimeout(context.Background(), time.Second*10) go func() { for { select { case <-ctx.Done(): fmt.Println("退出协程!!!") return default: fmt.Println("监控中 WithTimeout !!!") time.Sleep(1 * time.Second) } } }() time.Sleep(5 * time.Second) cancel() time.Sleep(2 * time.Second) fmt.Println("退出主函数!!!") }


![在这里插入图片描述](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/12013c70027d4f3db9a268776f0ac71b~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=x%2FbQmdz8wCKxOzn79xxLmNwAE%2FY%3D)


#### 5、使用context.WithDeadLine()方法


  使用 context.WithDeadLine() 方法,在指定的时间退出 goroutine。



func main() { nowTime := time.Now().Add(3 * time.Second) ctx, _ := context.WithDeadline(context.Background(), nowTime) go func() { for { select { case <-ctx.Done(): fmt.Println("退出协程!!!") return default: fmt.Println("监控中 WithDeadline !!!") time.Sleep(1 * time.Second) } } }() time.Sleep(5 * time.Second) fmt.Println("退出主函数!!!") }


![在这里插入图片描述](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/24cd4e5e86344622a3b678a8f8e5baef~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=VcCQofOz0J%2BGJx%2FNcAuaJa1bYZk%3D)


### 四、可以在 goroutine A 里去停止 goroutine B 吗?


  不可以,因为在 Go 语言中,goroutine 只能自己主动退出,一般通过 channel 来控制,不能被外界的其他 goroutine 关闭或干掉,也没有 goroutine 句柄的显式概念。goroutine 不可以跨 goroutine 强制停止。记住,在 Go 语言中每一个 goroutine 都需要自己承担自己的任何责任,这是基本原则。



> 
> 在 Go issues 中也有人提过类似问题,Dave Cheney 给出了一些思考:
> 
> 
> * 如果一个 goroutine 被强行停止了,它所拥有的资源会发生什么?堆栈被解开了吗?defer 是否被执行?  
>  如果执行 defer,该 goroutine 可能可以继续无限期地生存下去。  
>  如果不执行 defer,该 goroutine 原本的应用程序系统设计逻辑将会被破坏,这肯定不合理。
> * 如果允许强制停止 goroutine,是要释放所有东西,还是直接把它从调度器中踢出去,你想通过此解决什么问题?
> 
> 
> 这都是值得深思的,另外一旦放开这种限制,作为程序员,你维护代码,很有可能就不知道 goroutine 的句柄被传到了哪里,又是在何时何地被人莫名其妙关闭,非常糟糕。
> 
> 
> 



![img](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/4c071440bee04f37adefc50bc54a0c4a~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=nd6dUNq2d0x96U0ZG1%2BfPabHxow%3D)
![img](https://p9-xtjj-sign.byteimg.com/tos-cn-i-73owjymdk6/8177e80b56f94771af7ece1165fd39c6~tplv-73owjymdk6-jj-mark-v1:0:0:0:0:5o6Y6YeR5oqA5pyv56S-5Yy6IEAg5py65Zmo5a2m5Lmg5LmL5b-DQUk=:q75.awebp?rk3s=f64ab15b&x-expires=1771857316&x-signature=SgKCsZ6Uce46DI408z4uIzGATxM%3D)

**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化的资料的朋友,可以添加戳这里获取](https://gitee.com/vip204888)**


**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**