简单剖析tRPC-Go中使用的第三方协程池ants

tRPC-Go中的tRPC.Go()方法使用了ants协程池,做个简单剖析

panjf2000/ants协程池

在tRPC.Go方法(异步启动goroutine)中看到里面使用了ants协程池去实现(具体位置:g.pool.Invoke(p)

前置知识:

我们想异步完成一个任务,首先创建一个任务,然后需要从协程池(PoolWithFunc)中获取worker(goWorkerWithFunc),假设目前队列为空,这时一个worker和一个goroutine会一起创建出来,可以认为他俩就是绑一起的,然后处理完这个任务后,处于当前goroutine中的worker会放入全局的队列中,等待被其他协程去获取这个worker。

于是深入探索了下pool.Invoke等方法:

// github.com/panjf2000/ants/v2@v2.4.6/pool_func.go:162
// Invoke submits a task to pool.
func (p *PoolWithFunc) Invoke(args interface{}) error {// 这个func的caller:p就是一个协程池对象
	if p.IsClosed() {
		return ErrPoolClosed
	}
	var w *goWorkerWithFunc
	if w = p.retrieveWorker(); w == nil {// <----
		return ErrPoolOverload
	}
	w.args <- args // 这里展现了channel的异步通信能力:告知,具体可往下看**callback**
	return nil
}

首先执行p.retrieveWorker()获取一个worker,这个方法里面会根据当前worker队列的数量去做不同的逻辑:

  1. 若队列为空,表示无可用的worker,则需要新建(其中包含使用原生go去启动新协程)
// 会执行下面这个函数
spawnWorker := func() {
		w = p.workerCache.Get().(*goWorkerWithFunc)// workerCache是个sync.pool,从中获取goWorkerWithFunc对象
		w.run()// 启动一个 Goroutine 来重复执行传入的函数
	}
// worker 的结构如下:
// goWorkerWithFunc is the actual executor who runs the tasks,
// it starts a goroutine that accepts tasks and performs function calls.
type goWorkerWithFunc struct {
	// pool who owns this worker.
	pool *PoolWithFunc // 该 worker 所在协程池对象的指针
	// args is a job should be done.
	args chan interface{} // 待执行的任务
	// recycleTime will be update when putting a worker back into queue.
	recycleTime time.Time
}

为什么说这个Goroutine可以重复执行传入的函数?

答案:使用for循环不断获取无缓冲channel中的对象,获取不到则阻塞,直到管道关闭为止。

// run starts a goroutine to repeat the process
// that performs the function calls.
func (w *goWorkerWithFunc) run() {
	w.pool.incRunning()
	go func() {
		defer func() {
			// ...
		}()
		for args := range w.args {// w.args类型是chan any,用for循环不断获取无缓冲channel中的对象,获取不到则阻塞,直到管道关闭为止
			if args == nil {
				return
			}
			w.pool.poolFunc(args)// args对象就是每个传入的任务,poolFunc是这个协程池对象的执行任务的接口func
			if ok := w.pool.revertWorker(w); !ok {// 将该worker放入队列中,供其他人后续获取这个worker并利用当前这个协程去执行任务
				return
			}
		}
	}()
}

所以说:新开的这个协程会一直处于for循环中不断等待并执行新的任务。

callbackg.pool.Invoke(p)方法中的w.args <- args就是用来通知某个任务协程中的for args := range w.args去执行新的任务。

这里的p和args对象就是一个任务,结构如下:

p := &goerParam{
    ctx:     newCtx,// context.Context
    cancel:  cancel,// context.CancelFunc
    handler: handler,// func(context.Context) 调用方传入的闭包函数
}

sync.pool

上面可以看到 p.workerCache.Get().(*goWorkerWithFunc)中,任务p包含了一个workerCache属性,它是sync.pool类型,一个并发安全的对象池。说明worker都保存在一个对象池中,目的是减少内存分配和垃圾回收的开销。

有待深入…

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
'protoc-gen-go-trpc' 是一个用于生成 Go 语言的 gRPC 服务端和客户端代码的插件。它是基于 Google 的 Protocol Buffers(简称 Protobuf)和 gRPC 框架开发的。 Protobuf 是一种轻量级的数据序列化协议,可以用于结构化数据的序列化和反序列化。它定义了一种语言无关、平台无关的数据格式,可以用于不同语言之间的数据交换。gRPC 是一个高性能、开源的远程过程调用(RPC)框架,使用 Protobuf 作为默认的消息传输格式。 'protoc-gen-go-trpc' 插件是在 gRPC 的基础上扩展而来,它可以根据定义的 Protobuf 文件生成与 gRPC 相关的代码,包括服务端和客户端的接口定义、消息结构体、服务实现等。通过使用该插件,开发人员可以更方便地构建基于 gRPC 的分布式系统。 如果你遇到了 "'protoc-gen-go-trpc' 不是内部或外部命令,也不是可运行的程序" 的错误提示,可能是因为该插件没有正确安装或没有在系统的环境变量配置。你可以尝试以下步骤解决该问题: 1. 确保已正确安装 Go 编程语言,并且已配置好相关的环境变量。 2. 使用 Go 的包管理工具(如 go get)安装 'protoc-gen-go-trpc' 插件。可以执行以下命令进行安装: ``` go get github.com/tron-us/protobuf/protoc-gen-go-trpc ``` 3. 确保安装路径已添加到系统的环境变量。你可以将安装路径(一般为 $GOPATH/bin)添加到 PATH 环境变量。 安装和配置完成后,你就可以在命令行使用 'protoc-gen-go-trpc' 命令来生成 gRPC 相关的代码了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值