首页
技术小册
AIGC
面试刷题
技术文章
MAGENTO
云计算
视频课程
源码下载
PDF书籍
「涨薪秘籍」
登录
注册
上下文
上下文和普通参数的区别
上下文树
上下文接口——Context
利用context.emptyCtx创建树的根节点
上下文树的构建
利用valueCtx实现信息透传
valueCtx用于参数传递
从父节点获得透传值
利用cancelCtx通知协程终止执行
通知子协程终止执行
通知子协程的实现过程
为什么需要取消函数
利用timerCtx实现定时取消
调用context.WithDeadline()创建定时器上下文
调用context.WithTimeout()创建定时器上下文
编程范例——上下文的典型应用场景
利用结构体传递参数
valueContext为什么需要key
利用cancelCtx同时取消多个子协程
反射
反射的意义
反射的API
利用reflect.TypeOf()来获得类型信息
利用reflect.Type.Kind()方法来获取类型的具体分类
利用reflect.Type.Element()方法来获取元素类型
类型断言的用法与局限性
值信息
利用reflect.ValueOf()来获得值信息
利用reflect.Value.Kind()来获得值的分类信息
利用reflect.Value.Elem()来获得值的元素信息
利用反射访问和修改值信息
利用反射机制动态调用方法
编程范例——动态方法调用
泛型
泛型的意义
泛型应用到函数
泛型函数的使用
泛型中的隐含信息
避免类型强制转换
泛型类型的单独定义
泛型导致接口定义的变化
接口定义的变化
空接口的二义性
接口类型的限制
泛型类型应用到receiver
泛型类型不能直接用于定义receiver
间接实现泛型定义receiver
编程范例——自定义队列的实现
当前位置:
首页>>
技术小册>>
深入浅出Go语言核心编程(五)
小册名称:深入浅出Go语言核心编程(五)
### 章节:通知子协程的实现过程 在Go语言(Golang)的并发编程模型中,协程(Goroutine)是实现并发执行的关键组件。与传统的线程相比,Goroutine更加轻量级,能够高效地利用多核CPU资源,且其调度由Go运行时(runtime)自动管理,极大地简化了并发编程的复杂性。在实际应用中,经常需要在一个Goroutine中通知或等待另一个Goroutine的完成,以实现更加精细的并发控制。本章将深入探讨如何在Go语言中实现通知子协程的机制,包括使用通道(Channel)、WaitGroup、Context以及原子操作等方法。 #### 一、理解Goroutine与通道(Channel) 在Go中,通道(Channel)是实现Goroutine间通信的主要机制。通过通道,一个Goroutine可以发送数据到另一个Goroutine,实现数据的同步和共享。这种机制天然支持了Goroutine间的通知功能,因为发送操作可以视为一种通知,而接收操作则是对这种通知的响应。 ##### 示例:使用无缓冲通道通知 ```go package main import ( "fmt" "time" ) func worker(done chan bool) { fmt.Println("Working...") time.Sleep(time.Second) // 模拟耗时操作 fmt.Println("Done") // 发送通知 done <- true } func main() { done := make(chan bool) // 创建一个无缓冲通道 go worker(done) // 启动子协程 // 等待子协程的通知 <-done fmt.Println("Received done") } ``` 在上述示例中,`worker`函数执行完成后,通过向`done`通道发送一个布尔值来通知主Goroutine。主Goroutine通过从`done`通道接收值来等待`worker`函数的完成。 #### 二、使用WaitGroup等待多个子协程 当需要等待多个子Goroutine完成时,可以使用`sync.WaitGroup`。`WaitGroup`用于等待一组Goroutines的完成,它通过调用`Add`方法来增加等待的Goroutine数量,调用`Done`方法在Goroutine完成时减少计数,最后通过`Wait`方法阻塞,直到所有Goroutine都调用过`Done`。 ##### 示例:使用WaitGroup等待多个子协程 ```go package main import ( "fmt" "sync" "time" ) func worker(id int, wg *sync.WaitGroup) { defer wg.Done() // 确保在函数结束时调用Done fmt.Printf("Worker %d starting\n", id) time.Sleep(time.Second) // 模拟耗时操作 fmt.Printf("Worker %d done\n", id) } func main() { var wg sync.WaitGroup for i := 1; i <= 5; i++ { wg.Add(1) // 增加等待的Goroutine数量 go worker(i, &wg) // 启动子协程 } wg.Wait() // 等待所有子协程完成 fmt.Println("All workers have finished") } ``` #### 三、利用Context实现更灵活的协程控制 `context.Context`是Go标准库中的一个接口,它用于在不同Goroutine之间传递截止日期、取消信号以及其他请求范围的值。通过Context,可以更加灵活和强大地控制Goroutine的生命周期和行为。 ##### 示例:使用Context控制Goroutine ```go package main import ( "context" "fmt" "time" ) func worker(ctx context.Context, id int) { select { case <-time.After(2 * time.Second): fmt.Printf("Worker %d finished its task\n", id) case <-ctx.Done(): fmt.Printf("Worker %d cancelled\n", id) } } func main() { ctx, cancel := context.WithCancel(context.Background()) go worker(ctx, 1) time.Sleep(1 * time.Second) cancel() // 取消子协程 // 等待一段时间以确保看到结果(在实际应用中,可能需要更复杂的同步机制) time.Sleep(1 * time.Second) } ``` 在上面的例子中,我们创建了一个可取消的Context,并将其传递给`worker`函数。在`main`函数中,我们通过调用`cancel`函数来请求取消`worker`Goroutine。`worker`通过监听`ctx.Done()`通道来响应取消请求。 #### 四、原子操作与条件变量 虽然直接用于通知子协程的场景较少,但了解原子操作和条件变量对于深入理解并发控制机制仍然很重要。Go的`sync/atomic`包提供了底层的原子操作函数,如原子加、减、比较并交换等,这些操作在并发环境下是安全的,可以用来实现无锁的数据结构或同步机制。 条件变量通常与互斥锁(Mutex)结合使用,以在多个Goroutine间同步对共享资源的访问。虽然Go标准库没有直接提供条件变量的API,但可以通过通道、WaitGroup或Context等机制来实现类似的功能。 #### 五、总结 在Go语言中,实现通知子协程的机制多种多样,包括使用通道进行直接的通信和通知、使用WaitGroup等待多个子协程的完成、利用Context实现更灵活的控制逻辑,以及通过原子操作和互斥锁(间接方式)来实现更底层的同步控制。每种方法都有其适用场景和优缺点,开发者应根据实际需求选择合适的机制。 通过本章的学习,读者应该能够掌握在Go语言中实现子协程通知的基本方法和技巧,为编写高效、可靠的并发程序打下坚实的基础。
上一篇:
通知子协程终止执行
下一篇:
为什么需要取消函数
该分类下的相关小册推荐:
深入浅出Go语言核心编程(六)
Go语言入门实战经典
Go语言从入门到实战
go编程权威指南(四)
深入浅出Go语言核心编程(一)
WebRTC音视频开发实战
GO面试指南
企业级Go应用开发从零开始
深入浅出Go语言核心编程(三)
Golang修炼指南
深入浅出Go语言核心编程(八)
Go进阶之分布式爬虫实战