mirror of
https://gitee.com/johng/gf
synced 2026-06-26 17:35:40 +08:00
132 lines
4.0 KiB
Go
132 lines
4.0 KiB
Go
// Copyright 2017 gf Author(https://gitee.com/johng/gf). All Rights Reserved.
|
||
//
|
||
// This Source Code Form is subject to the terms of the MIT License.
|
||
// If a copy of the MIT was not distributed with this file,
|
||
// You can obtain one at https://gitee.com/johng/gf.
|
||
|
||
// Goroutine池.
|
||
// 用于goroutine复用,提升异步操作执行效率.
|
||
// 需要注意的是,grpool提供给的公共池不提供关闭方法(但可以修改公共属性),自创建的池可以手动关闭掉。
|
||
package grpool
|
||
|
||
import (
|
||
"math"
|
||
"runtime"
|
||
"sync/atomic"
|
||
"gitee.com/johng/gf/g/container/glist"
|
||
"errors"
|
||
)
|
||
|
||
const (
|
||
gDEFAULT_EXPIRE_TIME = 60 // 默认goroutine过期时间(秒)
|
||
gDEFAULT_CLEAR_INTERVAL = 60 // 定期检查任务过期时间间隔(秒)
|
||
)
|
||
|
||
// goroutine池对象
|
||
type Pool struct {
|
||
size int32 // 限制最大的goroutine数量/协程数/worker数量
|
||
expire int32 // goroutine过期时间(秒)
|
||
number int32 // 当前goroutine数量(非任务数)
|
||
queue *glist.List // 空闲任务队列(*PoolJob)
|
||
funcs *glist.List // 待处理任务操作队列
|
||
freeEvents chan struct{} // 空闲协程通知事件
|
||
funcEvents chan struct{} // 任务添加事件(兄弟们该干活了!)
|
||
stopEvents chan struct{} // 池关闭事件(用于池相关异步协程通知)
|
||
}
|
||
|
||
// goroutine worker
|
||
type PoolWorker struct {
|
||
job chan func() // 当前任务(当为nil时表示关闭)
|
||
pool *Pool // 所属协程池
|
||
update int64 // 更新时间
|
||
}
|
||
|
||
// 默认的goroutine池管理对象
|
||
// 该对象与进程同生命周期,无需Close
|
||
var defaultPool = New(gDEFAULT_EXPIRE_TIME)
|
||
|
||
// 创建goroutine池管理对象,给定过期时间(秒)
|
||
// 第二个参数用于限制限制最大的goroutine数量/线程数/worker数量,非必需参数,默认不做限制
|
||
func New(expire int, size...int) *Pool {
|
||
s := math.MaxInt32
|
||
if len(size) > 0 {
|
||
s = size[0]
|
||
}
|
||
p := &Pool {
|
||
size : int32(s),
|
||
expire : int32(expire),
|
||
queue : glist.New(),
|
||
funcs : glist.New(),
|
||
freeEvents : make(chan struct{}, math.MaxInt32),
|
||
funcEvents : make(chan struct{}, math.MaxInt32),
|
||
stopEvents : make(chan struct{}, runtime.GOMAXPROCS(-1) + 1),
|
||
}
|
||
p.startWorkLoop()
|
||
p.startClearLoop()
|
||
return p
|
||
}
|
||
|
||
// 添加异步任务(使用默认的池对象)
|
||
func Add(f func()) error {
|
||
return defaultPool.Add(f)
|
||
}
|
||
|
||
// 查询当前goroutine总数
|
||
func Size() int {
|
||
return int(atomic.LoadInt32(&defaultPool.number))
|
||
}
|
||
|
||
// 查询当前等待处理的任务总数
|
||
func Jobs() int {
|
||
return len(defaultPool.funcEvents)
|
||
}
|
||
|
||
// 动态改变默认池中goroutine的上线数量
|
||
func SetSize(size int) {
|
||
atomic.StoreInt32(&defaultPool.size, int32(size))
|
||
}
|
||
|
||
// 动态改变默认池中goroutine的过期时间
|
||
func SetExpire(expire int) {
|
||
atomic.StoreInt32(&defaultPool.expire, int32(expire))
|
||
}
|
||
|
||
// 添加异步任务
|
||
func (p *Pool) Add(f func()) error {
|
||
if len(p.stopEvents) > 0 {
|
||
return errors.New("pool closed")
|
||
}
|
||
p.funcs.PushBack(f)
|
||
p.funcEvents <- struct{}{}
|
||
return nil
|
||
}
|
||
|
||
// 查询当前goroutine worker总数
|
||
func (p *Pool) Size() int {
|
||
return int(atomic.LoadInt32(&p.number))
|
||
}
|
||
|
||
// 查询当前等待处理的任务总数
|
||
func (p *Pool) Jobs() int {
|
||
return len(p.funcEvents)
|
||
}
|
||
|
||
// 动态改变当前池中goroutine的上线数量
|
||
func (p *Pool) SetSize(size int) {
|
||
atomic.StoreInt32(&p.size, int32(size))
|
||
}
|
||
|
||
// 动态改变当前池中goroutine的过期时间
|
||
func (p *Pool) SetExpire(expire int) {
|
||
atomic.StoreInt32(&p.expire, int32(expire))
|
||
}
|
||
|
||
// 关闭池,所有的任务将会停止,此后继续添加的任务将不会被执行
|
||
func (p *Pool) Close() {
|
||
// 必须首先标识让任务过期自动关闭
|
||
p.SetExpire(-1)
|
||
// 使用stopEvents事件通知所有的异步协程及清理协程自动退出
|
||
for i := 0; i < runtime.GOMAXPROCS(-1) + 1; i++ {
|
||
p.stopEvents <- struct{}{}
|
||
}
|
||
} |