go-study/lock_free/delay_queue.go

80 lines
1.9 KiB
Go
Raw Normal View History

package lock_free
import (
"iter"
"sync"
"time"
)
2024-12-09 10:50:31 +08:00
type DelayLkQueue[TValue any, TKey comparable] struct {
timers map[TKey]*time.Timer
m sync.Mutex
2024-12-09 10:50:31 +08:00
LkQueue[TValue]
}
// NewDelayLkQueue 创建延迟无锁队列
2024-12-09 10:50:31 +08:00
func NewDelayLkQueue[TValue any, TKey comparable]() *DelayLkQueue[TValue, TKey] {
return &DelayLkQueue[TValue, TKey]{make(map[TKey]*time.Timer), sync.Mutex{}, *NewLkQueue[TValue]()}
}
// DelayEnqueue 延迟入队
2024-12-09 10:50:31 +08:00
func (q *DelayLkQueue[TValue, TKey]) DelayEnqueue(value TValue, duration time.Duration) {
time.AfterFunc(duration, func() {
q.Enqueue(value)
})
}
2024-12-09 10:50:31 +08:00
// CancellableDelayEnqueue 可取消的延迟入队
func (q *DelayLkQueue[TValue, TKey]) CancellableDelayEnqueue(key TKey, value TValue, duration time.Duration) {
if timer, ok := q.timers[key]; ok {
timer.Stop()
}
q.m.Lock()
defer q.m.Unlock()
2024-12-09 10:50:31 +08:00
q.timers[key] = time.AfterFunc(duration, func() {
delete(q.timers, key)
q.Enqueue(value)
})
}
// CancelDelayEnqueue 取消延迟入队
func (q *DelayLkQueue[TValue, TKey]) CancelDelayEnqueue(key TKey) {
q.m.Lock()
defer q.m.Unlock()
2024-12-09 10:50:31 +08:00
if timer, ok := q.timers[key]; ok {
delete(q.timers, key)
timer.Stop()
}
}
// ContinuousDequeue 持续监听出队
2024-12-09 10:50:31 +08:00
func (q *DelayLkQueue[TValue, TKey]) ContinuousDequeue() iter.Seq[TValue] {
return func(yield func(TValue) bool) {
for {
if value, ok := q.Dequeue(); ok {
if !yield(value) {
return
}
} else {
time.Sleep(time.Millisecond) // 队列为空休眠1毫秒
}
}
}
}
// ContinuousDequeueExecute 持续监听出队执行函数
2024-12-09 10:50:31 +08:00
func (q *DelayLkQueue[TValue, TKey]) ContinuousDequeueExecute(fn func(TValue)) {
for value := range q.ContinuousDequeue() {
fn(value)
}
}
// ContinuousDequeueNotify 持续监听出队通知
2024-12-09 10:50:31 +08:00
func (q *DelayLkQueue[TValue, TKey]) ContinuousDequeueNotify(chs ...chan TValue) {
q.ContinuousDequeueExecute(func(value TValue) {
for _, ch := range chs {
ch <- value
}
})
}