123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354 |
- package redis
- import (
- "sync/atomic"
- "time"
- )
- type rateLimiter struct {
- v int64
- _closed int64
- }
- func newRateLimiter(limit time.Duration, bucketSize int) *rateLimiter {
- rl := &rateLimiter{
- v: int64(bucketSize),
- }
- go rl.loop(limit, int64(bucketSize))
- return rl
- }
- func (rl *rateLimiter) loop(limit time.Duration, bucketSize int64) {
- for {
- if rl.closed() {
- break
- }
- if v := atomic.LoadInt64(&rl.v); v < bucketSize {
- atomic.AddInt64(&rl.v, 1)
- }
- time.Sleep(limit)
- }
- }
- func (rl *rateLimiter) Check() bool {
- for {
- if v := atomic.LoadInt64(&rl.v); v > 0 {
- if atomic.CompareAndSwapInt64(&rl.v, v, v-1) {
- return true
- }
- } else {
- return false
- }
- }
- }
- func (rl *rateLimiter) Close() error {
- atomic.StoreInt64(&rl._closed, 1)
- return nil
- }
- func (rl *rateLimiter) closed() bool {
- return atomic.LoadInt64(&rl._closed) == 1
- }
|