-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcounter.go
64 lines (57 loc) · 1.49 KB
/
counter.go
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
package percpu
import (
"sync/atomic"
)
// A Counter is an int64 counter which may be efficiently incremented
// by many goroutines concurrently.
//
// The counter is sharded into several CPU-local values which are written and
// read independently. Thus, the Load and Reset methods do not observe a
// consistent view of the total if they are called concurrently to Add.
//
// For example, suppose goroutine G1 runs
//
// counter.Add(1)
// counter.Add(2)
//
// and, concurrently, G2 runs
//
// t0 := counter.Reset()
// // wait for G1 to finish executing
// t1 := counter.Load()
//
// The value of t0 may be any of 0, 1, 2, or 3.
// The value of t1 may be any of 0, 1, 2, or 3 as well.
// However, t0+t1 must equal 3.
type Counter struct {
vs *Values // of *cval
}
type cval struct {
n int64
pad [120]byte // prevent false sharing
}
// NewCounter returns a fresh Counter initialized to zero.
func NewCounter() *Counter {
vs := NewValues(func() interface{} { return new(cval) })
return &Counter{vs: vs}
}
// Add adds n to the total count.
func (c *Counter) Add(n int64) {
atomic.AddInt64(&c.vs.Get().(*cval).n, n)
}
// Load computes the total counter value.
func (c *Counter) Load() int64 {
var sum int64
c.vs.Do(func(v interface{}) {
sum += atomic.LoadInt64(&v.(*cval).n)
})
return sum
}
// Reset sets the counter to zero and reports the old value.
func (c *Counter) Reset() int64 {
var sum int64
c.vs.Do(func(v interface{}) {
sum += atomic.SwapInt64(&v.(*cval).n, 0)
})
return sum
}