0%

go限制请求次数——令牌池

抓数据的网站限定1秒只能有10次请求,因此设计了一个令牌管理机制来控制请求数量。

设计思路如下:

  • 发请求前需要先获取令牌
  • 限定某时间段内的发放的令牌数量
  • 任务执行完成后不能归还令牌,只能使用定时器不断重置令牌
  • 如果当前goroutine数量过多时也不重置令牌
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
package main

import (
"errors"
"fmt"
"sync"
"sync/atomic"
"time"
)

//节流器

type throttle struct {
D time.Duration //周期是D,
C int64 //限制一个周期最多操作C次
Mu sync.Mutex
Token chan bool //令牌池
num int64 //当前的goroutine数量
maxNum int64 //允许工作goroutine最大数量
}

//如果两个周期后还没有申请到令牌,就报错超时
//目前用不到,如果限制routine最大数量需要靠这来监控
var ErrApplyTimeout = errors.New("apply token time out")

func NewThrottle(D time.Duration, C, maxNum int64) *throttle {
instance := &throttle{
D: D,
C: C,
Token: make(chan bool, C),
maxNum: maxNum,
}
go instance.reset()
return instance
}

//每周期重新填充一次令牌池
func (t *throttle) reset() {
ticker := time.NewTicker(t.D)
for _ = range ticker.C {
//goroutine数量不超过最大数量时再填充令牌池
if t.num >= t.maxNum {
continue
}
t.Mu.Lock()
supply := t.C - int64(len(t.Token))
fmt.Printf("reset token:%d\n", supply)
for supply > 0 {
t.Token <- true
supply--
}
t.Mu.Unlock()
}
}

//申请令牌,如果过两个周期还没申请到就报超时退出
func (t *throttle) ApplyToken() (bool, error) {
select {
case <-t.Token:
return true, nil
case <-time.After(t.D * 2):
return false, ErrApplyTimeout
}
}

func (t *throttle) Work(job func()) {
if ok, err := t.ApplyToken(); !ok {
fmt.Println(err)
return
}
go func() {
atomic.AddInt64(&t.num, 1)
defer atomic.AddInt64(&t.num, -1)
job()
}()
}
func main() {
t := NewThrottle(time.Second, 10, 20) //每秒10次,同时最多20个routine存在
for {
t.Work(doWork)
}
}

//真正的工作函数 假设每个需要执行5秒
func doWork() {
fmt.Println(time.Now())
<-time.After(5 * time.Second)
}