当前位置: 首页 > news >正文

golang 协程 如何中断和恢复

Go语言通知协程退出(取消)的几种方式 - 知乎

GoLang之goroutine底层系列二(goroutine的创建、让出、恢复)_golang goroutine-CSDN博客

在 Go 语言中,协程(也称为 goroutine)是通过 go 关键字启动的轻量级线程。由于 goroutine 的调度是由 Go 运行时管理的,直接停止一个正在执行的 goroutine 是不可能的,这与操作系统线程不同。但是,你可以通过一些策略来实现“中断”和“恢复”的行为,尽管这不是传统意义上的停止和恢复。

1. 使用通道(Channel)进行信号传递

最常用的方法是使用通道(channel)来安全地中断 goroutine。你可以在另一个 goroutine 中向该 goroutine 发送一个信号(例如,一个特殊的值或者一个结构体),然后在被中断的 goroutine 中定期检查这个通道以决定是否退出循环或函数。

package mainimport ("fmt""time"
)func worker(done chan bool) {for {select {case <-done:fmt.Println("Worker stopped.")return // 退出函数,结束 goroutinedefault:fmt.Println("Working...")time.Sleep(500 * time.Millisecond) // 模拟工作}}
}func main() {done := make(chan bool)go worker(done)time.Sleep(2 * time.Second) // 等待一段时间done <- true               // 发送信号到 worker,要求它停止time.Sleep(1 * time.Second) // 等待 worker 完全停止
}

2. 使用 context 包

context 包是 Go 标准库中处理请求范围的值、取消信号和截止时间的标准方式。它非常适合用来控制 goroutine 的生命周期。

示例代码:

package mainimport ("context""fmt""time"
)func worker(ctx context.Context) {for {select {case <-ctx.Done(): // 当 ctx 被取消时,这里会被触发fmt.Println("Worker stopped.")return // 退出函数,结束 goroutinedefault:fmt.Println("Working...")time.Sleep(500 * time.Millisecond) // 模拟工作}}
}func main() {ctx, cancel := context.WithCancel(context.Background())go worker(ctx)time.Sleep(2 * time.Second) // 等待一段时间cancel()                    // 取消 context,要求 worker 停止time.Sleep(1 * time.Second) // 等待 worker 完全停止
}

3. 使用 sync.Once 或 sync.Mutex 控制退出逻辑的执行一次

如果你需要在某些情况下只安全地执行退出逻辑一次,可以使用 sync.Once 来确保。这对于确保资源清理代码只运行一次非常有用。但是,这种方法更多地是关于如何安全地执行退出逻辑,而不是真正地“中断”一个正在执行的 goroutine。

示例代码:

package mainimport ("fmt""sync""time"
)var once sync.Once
var running = true // 控制循环是否继续运行的标志位func worker() {for running { // 使用 running 作为循环条件来控制何时退出循环体中的代码块。fmt.Println("Working...")time.Sleep(500 * time.Millisecond) // 模拟工作}
}func stopWorker() { // 安全地停止 worker 的函数。使用 sync.Once 确保只调用一次。once.Do(func() { // 使用 sync.Once 来确保这个操作只执行一次。running = false // 设置 running 为 false,让循环在下次迭代时退出。})
}

在上面的代码中,你可以通过调用 stopWorker() 来停止 worker() 函数。但是,这种方法更多地是控制何时退出循环而不是真正地“中断”一个正在执行的 goroutine。在实践中,通常推荐使用通道或 context 来管理 goroutine 的生命周期。

http://www.dtcms.com/a/268356.html

相关文章:

  • WHAT - 依赖管理工具 CocoaPods
  • 从小白到进阶:解锁linux与c语言高级编程知识点嵌入式开发的任督二脉(1)
  • 如何确保Kafka集群的高可用?
  • 【MySQL】DTS机制对触发器时间的影响
  • Python-可视化学习笔记
  • 【机器学习笔记Ⅰ】3 代价函数
  • 空调和烘干机的使用
  • pyhton基础【23】面向对象进阶四
  • 爬虫的笔记整理
  • 在Ubuntu 24.04上部署Zabbix 7.0对服务器进行监控
  • Grok 4 最新技术评测与发布指南
  • 位置编码和RoPE
  • 光纤的最小弯曲半径是多少?
  • 商业秘密攻防战:技术信息与经营信息的界定之道
  • 基于Flask和机器学习开发的米其林餐厅数据可视化平台
  • 爬虫-request模块使用
  • CSS05:结构伪类选择器和属性选择器
  • 反向遍历--当你修改一个元素的outerHTML时,该元素会被从 DOM 中移除
  • 大模型RLHF中PPO强化学习代码学习笔记(二)
  • 回环检测 Scan Contex
  • DolphinScheduler 3.2.0 后端开发环境搭建指南
  • XML 笔记
  • 极简的神经网络反向传播例子
  • 用户中心Vue3项目开发2.0
  • Docker 容器编排原理与使用详解
  • 125.【C语言】数据结构之归并排序递归解法
  • FileZilla二次开发实战指南:C++架构解析与界面功能扩展
  • 操作系统王道考研习题
  • 76、覆盖最小子串
  • 【STM32】通用定时器PWM