golang并发控制

常见的并发控制
常见的并发控制

channel:通过无缓冲的channel进行同步调用,有缓冲的channel进行异步调用,也可限制并发数量

waitgroup:可以通过add来动态调整,释放的时间需要使用defer 进行wg.done操作

context:通过在协程之间传递上下文消息,可对派生的子协程进行并发控制,防止资源浪费

高并发任务控制包

b站开源的并发控制包,对以上三种并发控制包的优点进行融合。限制并发数量,对并发上下文控制等。 虽然官方在go1.17版本也引入并发控制包,但没有b站开源的支持场景多,建议使用b站开源包,简单场景可使用官方包。我们一般使用高并发,处理大量的数据分析工作。对用一个job拥有子任务的时候,对于整个任务有两种限制:

  • 一个任务失败会导致所有*未进行或进行中*的任务被 cancel
  • 一个任务失败导致所有任务被 cancel

Notice:在并行进行任务调度的时候,做好业务侧兜底,失败的任务进行日志记录,告警等。

并发控制包

kratos并发控制包:

errgroup package - github.com/bilibili/kratos/pkg/sync/errgroup - Go Packages

官方1.17后引入:

https://pkg.go.dev/golang.org/x/sync/errgroup#pkg-overview

b站开源开发控制包demo详情

并发数量限制

设置最大并行数 GOMAXPROCS 对以上三种使用方式均起效 NOTE: 由于 errgroup 实现问题,设定 GOMAXPROCS 的 errgroup 需要立即调用 Wait() 例如:

复制代码
g := errgroup.WithCancel(ctx)
g.GOMAXPROCS(2)
// task1
g.Go(func(ctx context.Context) {
    fmt.Println("task1")
})
// task2
g.Go(func(ctx context.Context) {
    fmt.Println("task2")
})
// task3
g.Go(func(ctx context.Context) {
    fmt.Println("task3")
})
// NOTE: 此时设置的 GOMAXPROCS 为2, 添加了三个任务 task1, task2, task3 此时 task3 是不会运行的!
// 只有调用了 Wait task3 才有运行的机会
g.Wait() // task3 运行

子任务失败所有任务被取消

复制代码
g := errgroup.WithCancel(ctx)
g.Go(func(ctx context.Context) {
    // NOTE: 此时 ctx 是从 errgroup.WithContext 传递的 ctx 派生出的 ctx
    // do something
})

子任务执行失败不影响其他子任务继续执行

复制代码
直接使用 此时不会因为一个任务失败导致所有任务被 cancel:

g := &errgroup.Group{}
g.Go(func(ctx context.Context) {
	// NOTE: 此时 ctx 为 context.Background()
	// do something
})
WithContext 使用 WithContext 时不会因为一个任务失败导致所有任务被 cancel:

g := errgroup.WithContext(ctx)
g.Go(func(ctx context.Context) {
	// NOTE: 此时 ctx 为 errgroup.WithContext 传递的 ctx
	// do something
})

参考资料:

errgroup package - github.com/bilibili/kratos/pkg/sync/errgroup - Go Packages

errgroup package - golang.org/x/sync/errgroup - Go Packages

相关推荐
zopple2 小时前
常见的 Spring 项目目录结构
java·后端·spring
是娇娇公主~3 小时前
C++ 中 std::deque 的原理?它内部是如何实现的?
开发语言·c++·stl
SuperEugene3 小时前
Axios 接口请求规范实战:请求参数 / 响应处理 / 异常兜底,避坑中后台 API 调用混乱|API 与异步请求规范篇
开发语言·前端·javascript·vue.js·前端框架·axios
lars_lhuan4 小时前
Go WaitGroup 源码解析
golang
xuxie994 小时前
N11 ARM-irq
java·开发语言
cjy0001114 小时前
springboot的 nacos 配置获取不到导致启动失败及日志不输出问题
java·spring boot·后端
wefly20175 小时前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
小江的记录本5 小时前
【事务】Spring Framework核心——事务管理:ACID特性、隔离级别、传播行为、@Transactional底层原理、失效场景
java·数据库·分布式·后端·sql·spring·面试
sheji34165 小时前
【开题答辩全过程】以 基于springboot的校园失物招领系统为例,包含答辩的问题和答案
java·spring boot·后端
luanma1509805 小时前
PHP vs C++:编程语言终极对决
开发语言·c++·php