@@ -28,36 +28,115 @@ session的基本原理是服务端为每一个session维护一份会话信息数
28
28
我们就来定义一个全局的session管理器
29
29
30
30
type SessionManager struct {
31
- CookieName string
32
- lock sync.Mutex
33
- sessions map[string]map[string]interface{}
34
- list list.List
31
+ cookieName string //识别cookie
32
+ lock sync.Mutex //用来锁
33
+ sessions map[string]*session //用来存储
34
+ list * list.List //用来做gc
35
35
}
36
36
37
- Go实现整个的流程应该也是这样的,创建一个全部的session管理器
37
+ Go实现整个的流程应该也是这样的,在main包中创建一个全部的session管理器
38
38
39
- globalSessions = NewSessions()
39
+ var globalSessions *SessionManager
40
+ //然后在init函数中初始化
41
+ func init() {
42
+ globalSessions = NewSessions()
43
+ }
44
+
45
+ 接下来的例子中创建的session都是放在内存里面的,当然也可以放在存储中,这个后面会详细的讲解。
40
46
41
- 我们接下来的例子创建的session放在内存里面,当然也可以放在存储中,这个后面会详细的讲解。
47
+ 接下来我们来定义针对每个用户的session对象
48
+
49
+ type session strcut{
50
+ sid string //session id唯一标示
51
+ timeAccessed time.Time //最后访问时间
52
+ value map[string]interface{} //session里面存储的值
53
+ }
42
54
43
55
###创建
44
56
首先我们要产生一个唯一的sessionid
45
57
46
- func sessionId() string {
58
+ func (this *SessionManager) sessionId() string {
47
59
var n int = 24
48
60
b := make([]byte, n)
49
61
io.ReadFull(rand.Reader, b)
50
62
return base64.URLEncoding.EncodeToString(b)
51
63
}
52
64
53
- 那么这个sessionid什么时候创建呢?
65
+ 那么这个sessionid什么时候创建呢?我们在启动我们的处理器的时候来进行session的创建,这里我继续前面表单处理的例子说明:
66
+
67
+ func login(w http.ResponseWriter, r *http.Request) {
68
+
69
+ fmt.Println("method:", r.Method) //获取请求的方法
70
+ if r.Method == "GET" {
71
+ t, _ := template.ParseFiles("login.gtpl")
72
+ t.Execute(w, nil)
73
+ } else {
74
+ //请求的是登陆数据,那么执行登陆的逻辑判断
75
+ fmt.Println("username:", r.Form["username"])
76
+ fmt.Println("password:", r.Form["password"])
77
+ }
78
+ }
79
+
80
+ //开始一个新的sessionid
81
+ //先通过cookiename查询是否已经中了sessionid,然后查找这个sessionid是否还在sessions里面
82
+ //如果还在那么就更新timeAccessed,如果不存在就addnew一个
83
+ func (this *SessionManager) SessionStart() {
84
+
85
+ }
86
+
87
+ //销毁当前sessionid
88
+ func (this *SessionManager) SessionDestroy() {
89
+
90
+ }
91
+
92
+ //新增一个session
93
+ func (this *SessionManager) AddNew(key string, value map[string]interface{}) {
94
+ newsid :=this.sessionId()
95
+ newsess:=&session{"sid":newsid,"timeAccessed":time.Now(),"value":value}
96
+ this.sessions[key]=newsess
97
+ this.list.Push(newsess)
98
+ }
54
99
55
100
###保持
56
101
57
- ###设置值
102
+
103
+ ###操作值:设置、读取和删除
104
+ session对象需要有设置和删除操作
105
+
106
+ func (this *session) Set(key string, value interface{}){
107
+
108
+ }
109
+
110
+ func (this *session) Get(key string){
111
+
112
+ }
113
+
114
+ func (this *session) Del(key string){
115
+
116
+ }
117
+
118
+ 当我们设置完值之后需要去更新相应的sessionmanager里面的值
119
+
120
+ func (this *SessionManager) update(element *list.Element, value map[string]interface{}) {
121
+ this.lock.Lock()
122
+ defer this.lock.Unlock()
123
+ element.Value.(*session).value = value
124
+ this.moveToFront(element)
125
+ }
126
+
127
+ ###如何实时更新时间
128
+ 我们的session是有一定的时间限制的,那么如果用户一直在访问页面,那么我们需要实时的去更新这个时间,如何来做这一步呢?请看下面的函数处理
129
+
130
+ func (this *SessionManager) moveToFront(element *list.Element) {
131
+ this.lock.Lock()
132
+ defer this.lock.Unlock()
133
+ this.list.MoveToFront(element)
134
+ element.Value.(*session).timeAccessed = time.Now()
135
+ }
136
+
58
137
59
138
###销毁
60
- 我们来看一下Go如何来管理销毁
139
+ 我们来看一下Go如何来管理销毁,我们启动一个单独的goroutine来做session的销毁:
61
140
62
141
func GC() {
63
142
for true {
@@ -71,6 +150,26 @@ Go实现整个的流程应该也是这样的,创建一个全部的session管
71
150
}
72
151
73
152
153
+ func (this *SessionManager) gc() {
154
+ this.lock.Lock()
155
+ defer this.lock.Unlock()
156
+ var expire int64 = 60 * 15 // 15 minutes
157
+ // back to front
158
+ for {
159
+ element := this.list.Back()
160
+ if element == nil {
161
+ break
162
+ }
163
+ if (element.Value.(*session).timeAccessed.Unix() + expire) < time.Now().Unix() {
164
+ this.list.Remove(element)
165
+ delete(this.table, element.Value.(*session).sid)
166
+ } else {
167
+ break
168
+ }
169
+ }
170
+ }
171
+
172
+
74
173
75
174
##总结
76
175
0 commit comments