-
Notifications
You must be signed in to change notification settings - Fork 25
/
thread_test.go
120 lines (113 loc) · 3.05 KB
/
thread_test.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
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
package routine
import (
"bytes"
"context"
"runtime"
"runtime/pprof"
"sync"
"testing"
"time"
"github.com/stretchr/testify/assert"
)
func TestCurrentThread(t *testing.T) {
assert.NotNil(t, currentThread(true))
assert.Same(t, currentThread(true), currentThread(true))
}
func TestPProf(t *testing.T) {
const concurrency = 10
const loopTimes = 10
tls := NewThreadLocal[any]()
tls.Set("你好")
wg := &sync.WaitGroup{}
wg.Add(concurrency)
for i := 0; i < concurrency; i++ {
tmp := i
go func() {
for j := 0; j < loopTimes; j++ {
time.Sleep(100 * time.Millisecond)
tls.Set(tmp)
assert.Equal(t, tmp, tls.Get())
pprof.Do(context.Background(), pprof.Labels("key", "value"), func(ctx context.Context) {
assert.Nil(t, currentThread(false))
assert.Nil(t, tls.Get())
tls.Set("hi")
//
label, find := pprof.Label(ctx, "key")
assert.True(t, find)
assert.Equal(t, "value", label)
//
assert.Equal(t, "hi", tls.Get())
//
label2, find2 := pprof.Label(ctx, "key")
assert.True(t, find2)
assert.Equal(t, "value", label2)
})
assert.Nil(t, tls.Get())
}
wg.Done()
}()
}
assert.Nil(t, pprof.StartCPUProfile(&bytes.Buffer{}))
wg.Wait()
pprof.StopCPUProfile()
assert.Equal(t, "你好", tls.Get())
}
func TestThreadGC(t *testing.T) {
const allocSize = 10_000_000
tls := NewThreadLocal[[]byte]()
tls2 := NewInheritableThreadLocal[[]byte]()
allocWait := &sync.WaitGroup{}
allocWait.Add(1)
gatherWait := &sync.WaitGroup{}
gatherWait.Add(1)
gcWait := &sync.WaitGroup{}
gcWait.Add(1)
//=========Init
heapInit, numInit := getMemStats()
printMemStats("Init", heapInit, numInit)
//
task := GoWait(func(token CancelToken) {
tls.Set(make([]byte, allocSize))
tls2.Set(make([]byte, allocSize))
go func() {
gcWait.Wait()
}()
task2 := GoWaitResult(func(token CancelToken) int {
return 1
})
assert.Equal(t, 1, task2.Get())
allocWait.Done() //alloc ok, release main thread
gatherWait.Wait() //wait gather heap info
})
//=========Alloc
allocWait.Wait() //wait alloc done
heapAlloc, numAlloc := getMemStats()
printMemStats("Alloc", heapAlloc, numAlloc)
assert.Greater(t, heapAlloc, heapInit+allocSize*2*0.9)
assert.Greater(t, numAlloc, numInit)
//=========GC
gatherWait.Done() //gather ok, release sub thread
task.Get() //wait sub thread finish
time.Sleep(500 * time.Millisecond)
heapGC, numGC := getMemStats()
printMemStats("AfterGC", heapGC, numGC)
gcWait.Done()
//=========Summary
heapRelease := heapAlloc - heapGC
numRelease := numAlloc - numGC
printMemStats("Summary", heapRelease, numRelease)
assert.Greater(t, int(heapRelease), int(allocSize*2*0.9))
assert.Equal(t, 1, numRelease)
}
func getMemStats() (uint64, int) {
stats := runtime.MemStats{}
runtime.GC()
runtime.ReadMemStats(&stats)
return stats.HeapAlloc, runtime.NumGoroutine()
}
func printMemStats(section string, heapAlloc uint64, numGoroutine int) {
//fmt.Printf("%v\n", section)
//fmt.Printf("HeapAlloc = %v\n", heapAlloc)
//fmt.Printf("NumGoroutine = %v\n", numGoroutine)
//fmt.Printf("===\n")
}