-
Notifications
You must be signed in to change notification settings - Fork 105
/
Copy pathSessionIDManager.go
91 lines (71 loc) · 2.07 KB
/
SessionIDManager.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
package main
import (
"sync"
"github.com/bits-and-blooms/bitset"
)
//////////////////////////////// SessionIDManager //////////////////////////////
// SessionIDManager 线程安全的会话ID管理器
type SessionIDManager struct {
lock sync.Mutex
sessionIDs *bitset.BitSet
count uint16 // how many ids are used now
allocIDx uint16
maxSessionId uint16 // sessionID可以达到的最大数值
}
// NewSessionIDManager 创建一个会话ID管理器实例
func NewSessionIDManager(maxSessionId uint16) (manager *SessionIDManager, err error) {
manager = new(SessionIDManager)
manager.maxSessionId = maxSessionId
manager.sessionIDs = bitset.New(uint(manager.maxSessionId + 1))
manager.count = 0
manager.sessionIDs.ClearAll()
return
}
// isFull 判断会话ID是否已满(内部使用,不加锁)
func (manager *SessionIDManager) isFullWithoutLock() bool {
return (manager.count > manager.maxSessionId)
}
// IsFull 判断会话ID是否已满
func (manager *SessionIDManager) IsFull() bool {
defer manager.lock.Unlock()
manager.lock.Lock()
return manager.isFullWithoutLock()
}
func (manager *SessionIDManager) next() {
manager.allocIDx++
if manager.allocIDx > manager.maxSessionId {
manager.allocIDx = 0
}
}
// AllocSessionID 为调用者分配一个会话ID
func (manager *SessionIDManager) AllocSessionID() (sessionID uint16, err error) {
defer manager.lock.Unlock()
manager.lock.Lock()
if manager.isFullWithoutLock() {
sessionID = manager.maxSessionId
err = ErrSessionIDFull
return
}
// find an empty bit
for manager.sessionIDs.Test(uint(manager.allocIDx)) {
manager.next()
}
// set to true
manager.sessionIDs.Set(uint(manager.allocIDx))
manager.count++
sessionID = manager.allocIDx
err = nil
manager.next()
return
}
// FreeSessionID 释放调用者持有的会话ID
func (manager *SessionIDManager) FreeSessionID(sessionID uint16) {
defer manager.lock.Unlock()
manager.lock.Lock()
if !manager.sessionIDs.Test(uint(sessionID)) {
// ID未分配,无需释放
return
}
manager.sessionIDs.Clear(uint(sessionID))
manager.count--
}