vRp.CD2g_test/utils/timer/ticker.go

132 lines
3.2 KiB
Go

package timer
import (
"github.com/gorhill/cronexpr"
"reflect"
"sync"
"time"
"github.com/RussellLuo/timingwheel"
)
// Ticker 定时器
type Ticker struct {
timer *Timer
wheel *timingwheel.TimingWheel
timers map[string]*Scheduler
lock sync.RWMutex
handle func(name string, caller func())
mark string
}
// Mark 获取定时器的标记
// - 通常用于鉴别定时器来源
func (slf *Ticker) Mark() string {
return slf.mark
}
// Release 释放定时器,并将定时器重新放回 Timer 池中
func (slf *Ticker) Release() {
slf.timer.lock.Lock()
defer slf.timer.lock.Unlock()
slf.lock.Lock()
slf.mark = ""
for name, scheduler := range slf.timers {
scheduler.close()
delete(slf.timers, name)
}
slf.lock.Unlock()
slf.timer.tickers = append(slf.timer.tickers, slf)
}
// StopTimer 停止特定名称的调度器
func (slf *Ticker) StopTimer(name string) {
slf.lock.Lock()
defer slf.lock.Unlock()
if s, ok := slf.timers[name]; ok {
s.close()
delete(slf.timers, name)
}
}
// IsStopped 特定名称的调度器是否已停止
func (slf *Ticker) IsStopped(name string) bool {
slf.lock.RLock()
defer slf.lock.RUnlock()
if s, ok := slf.timers[name]; ok {
return s.isClosed()
}
return true
}
// GetSchedulers 获取所有调度器名称
func (slf *Ticker) GetSchedulers() []string {
slf.lock.RLock()
defer slf.lock.RUnlock()
names := make([]string, 0, len(slf.timers))
for name := range slf.timers {
names = append(names, name)
}
return names
}
// Cron 通过 cron 表达式设置一个调度器,当 cron 表达式错误时,将会引发 panic
func (slf *Ticker) Cron(name, expression string, handleFunc interface{}, args ...interface{}) {
expr := cronexpr.MustParse(expression)
slf.loop(name, 0, 0, expr, 0, handleFunc, args...)
}
// After 设置一个在特定时间后运行一次的调度器
func (slf *Ticker) After(name string, after time.Duration, handleFunc interface{}, args ...interface{}) {
slf.loop(name, after, timingWheelTick, nil, 1, handleFunc, args...)
}
// Loop 设置一个在特定时间后反复运行的调度器
func (slf *Ticker) Loop(name string, after, interval time.Duration, times int, handleFunc interface{}, args ...interface{}) {
slf.loop(name, after, interval, nil, times, handleFunc, args...)
}
// Loop 设置一个在特定时间后反复运行的调度器
func (slf *Ticker) loop(name string, after, interval time.Duration, expr *cronexpr.Expression, times int, handleFunc interface{}, args ...interface{}) {
slf.StopTimer(name)
if expr == nil {
if after < timingWheelTick {
after = timingWheelTick
}
if interval < timingWheelTick {
interval = timingWheelTick
}
}
var values = make([]reflect.Value, len(args))
for i, v := range args {
values[i] = reflect.ValueOf(v)
}
scheduler := &Scheduler{
name: name,
after: after,
interval: interval,
total: times,
cbFunc: reflect.ValueOf(handleFunc),
cbArgs: values,
ticker: slf,
expr: expr,
}
slf.lock.Lock()
slf.timers[name] = scheduler
if slf.handle != nil {
scheduler.timer = slf.wheel.ScheduleFunc(scheduler, func() {
slf.handle(scheduler.Name(), scheduler.Caller)
})
} else {
scheduler.timer = slf.wheel.ScheduleFunc(scheduler, scheduler.Caller)
}
slf.lock.Unlock()
}