前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Context:goroutines编排(二)

Context:goroutines编排(二)

原创
作者头像
rxg456
发布2022-05-25 02:09:20
6060
发布2022-05-25 02:09:20
举报
文章被收录于专栏:野生云+

1、介绍

项目地址:https://github.com/oklog/run prometheus就是使用这种方式管理多goroutine编排

run.Group是一种管理goroutine组件生命周期的通用机制,它在任何需要将多个goroutines作为一个单元整体进行协调的情况下都很有用。

2、使用

创建一个零值的run.Group,然后向其添加actors。actor被定义为一对函数:一个执行函数,它应该同步运行;一个中断函数,当被调用时,它应该使执行函数返回。最后,调用Run,它同时运行所有的组件,等待直到第一个actor退出,调用中断函数,最后只有在所有组件都返回后才将控制权返回给调用者。这个通用的API允许调用者对几乎所有可运行的任务进行编排,并为组件实现定义良好的生命周期管理。

2.1 简单使用

  • g.add 第一个参数为run函数,返回error,要求该函数长时间运行,遇到错误再退出
    • 具体场景有 for +ticker +ctx
    • 长时间运行的 http 和rpc 用error chan 在g.add中做
  • 第二个参数为interrupt函数,作用是退出时做一些清理操作
代码语言:javascript
复制
import (
  "context"
  "fmt"
  "os"
  "os/signal"
  "syscall"
​
  "github.com/oklog/run"
)
​
func main() {
  // 编排开始
  var g run.Group
  ctxAll, cancelAll := context.WithCancel(context.Background())
  fmt.Println(ctxAll)
  {
    // 处理信号退出的channel
    term := make(chan os.Signal, 1)
​
    signal.Notify(term, os.Interrupt, syscall.SIGTERM)
    cancelC := make(chan struct{})
    g.Add(
      func() error {
        select {
        case <-term:
          fmt.Println("Receive SIGTERM, exiting gracefully...")
          cancelAll()
          return nil
        case <-cancelC:
          fmt.Println("other cancel exiting")
          return nil
        }
      },
      func(err error) {
        close(cancelC)
      },
    )
  }
  g.Run()
​
}

2.2 进阶使用

主协程退出触发cancelAll(),通知ctxAll的所有协程触发ctxAll.Done()退出

代码语言:javascript
复制
import (
  "context"
  "fmt"
  "os"
  "os/signal"
  "syscall"
  "time"
​
  "github.com/oklog/run"
)
​
func main() {
  // 编排开始
  var g run.Group
  ctxAll, cancelAll := context.WithCancel(context.Background())
  {
    // 处理信号退出的channel
    term := make(chan os.Signal, 1)
​
    signal.Notify(term, os.Interrupt, syscall.SIGTERM)
    cancelC := make(chan struct{})
    g.Add(
      func() error {
        select {
        case <-term:
          fmt.Println("Receive SIGTERM, exiting gracefully...")
          cancelAll()
          return nil
        case <-cancelC:
          fmt.Println("other cancel exiting")
          return nil
        }
      },
      func(err error) {
        close(cancelC)
      },
    )
  }
  {
    g.Add(func() error {
      for {
        ticker := time.NewTicker(3 * time.Second)
        select {
        case <-ctxAll.Done():
          fmt.Println("打工人01,接收到了cancelAll的退出指令")
          return nil
        case <-ticker.C:
          fmt.Println("我是打工人01")
​
        }
      }
    }, func(err error) {
​
    },
    )
  }
  g.Run()
​
}

3、源码分析

3.1 run.Group

  • actor的结构体

3.2 Add

  • 向actor组添加(Add)一个函数,每个actor必须可以被一个interrupt中断函数抢占。也就是说,如果interrupt中断被调用,execute应该返回。而且,即使在execute返回之后,调用interrupt也必须是安全的。
  • 第一个返回的actor(函数)会中断所有正在运行的actors
  • 错误被传递给中断函数,并由Run返回

3.3 Run

概述:同时运行所有的actors函数,当第一个actor返回时,所有其他的actors都会被打断。只有当所有的actors都退出时,Run才会返回。运行返回第一个退出的actor所返回的错误

细节

  • 判断是否有actors,就是有没有Add
  • len运行每一个actors
  • 等待第一个actor停止
  • 向所有actors发出interrupt(中断)信号
  • 等待所有actors都停止下来,有些快有些慢所以用channel
  • 返回第一个actor触发的错误

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1、介绍
  • 2、使用
    • 2.1 简单使用
      • 2.2 进阶使用
      • 3、源码分析
        • 3.1 run.Group
          • 3.2 Add
            • 3.3 Run
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档