如何使用 Go http 包提供共享结构?
How to serve shared struct with Go http package?
我有一个包含许多字段的结构(其中一些字段也是指向其他结构的指针),这些字段在单独的 goroutine 中不断更新。当提供页面时,从 go 的 http
模板访问相同的结构。
代码示例:
type SharedStruct struct {
Description string
Counter int
Status_ *Status
LastChecked time.Time
//other fields
}
var shared = &SharedStruct{}
go func() {
//..updates fields every 5 minutes
}()
go-http 处理程序:
func someHandler(w http.ResponseWriter, r *http.Request) {
t.ExecuteTemplate(w, "page.html", shared)
}
和page.html
模板:
...
Status: {{.Status_.StatusCode}}
Counter: {{.Counter}}
Last checked: {{.LastChecked.Format "2006-02-01 15:04:05"}}
到目前为止,一切都按预期进行,但我知道如果没有任何同步,可能会发生不好的事情。正确处理此问题的首选方法是什么?
首选方式与任何其他情况相同。
在读取/更新共享结构时使用互斥锁:
var shared = &SharedStruct{}
var mux = &sync.RWMutex{}
func someHandler(w http.ResponseWriter, r *http.Request) {
mux.RLock()
defer mux.RUnlock()
t.ExecuteTemplate(w, "page.html", shared)
}
// Code that modifies shared:
mux.Lock()
shared.Counter++
mux.Unlock()
或者如果模板执行需要很长时间,复制 shared
结构并在执行模板时传递副本可能是有利的,这样在模板执行期间访问 shared
没有被屏蔽。请注意,在制作副本时,您仍然必须使用互斥锁。此外,如果不仅指针而且指向的值可能会发生变化,您还必须复制这些:
func someHandler(w http.ResponseWriter, r *http.Request) {
mux.RLock()
shared2 := &SharedStruct{}
*shared2 = *shared
shared2.Status_ = new(Status)
*shared2.Status_ = *shared.Status_
mux.RUnlock()
t.ExecuteTemplate(w, "page.html", shared2)
}
如果模板只使用 shared
字段的一小部分,当然只复制这些字段就足够了。
我有一个包含许多字段的结构(其中一些字段也是指向其他结构的指针),这些字段在单独的 goroutine 中不断更新。当提供页面时,从 go 的 http
模板访问相同的结构。
代码示例:
type SharedStruct struct {
Description string
Counter int
Status_ *Status
LastChecked time.Time
//other fields
}
var shared = &SharedStruct{}
go func() {
//..updates fields every 5 minutes
}()
go-http 处理程序:
func someHandler(w http.ResponseWriter, r *http.Request) {
t.ExecuteTemplate(w, "page.html", shared)
}
和page.html
模板:
...
Status: {{.Status_.StatusCode}}
Counter: {{.Counter}}
Last checked: {{.LastChecked.Format "2006-02-01 15:04:05"}}
到目前为止,一切都按预期进行,但我知道如果没有任何同步,可能会发生不好的事情。正确处理此问题的首选方法是什么?
首选方式与任何其他情况相同。
在读取/更新共享结构时使用互斥锁:
var shared = &SharedStruct{}
var mux = &sync.RWMutex{}
func someHandler(w http.ResponseWriter, r *http.Request) {
mux.RLock()
defer mux.RUnlock()
t.ExecuteTemplate(w, "page.html", shared)
}
// Code that modifies shared:
mux.Lock()
shared.Counter++
mux.Unlock()
或者如果模板执行需要很长时间,复制 shared
结构并在执行模板时传递副本可能是有利的,这样在模板执行期间访问 shared
没有被屏蔽。请注意,在制作副本时,您仍然必须使用互斥锁。此外,如果不仅指针而且指向的值可能会发生变化,您还必须复制这些:
func someHandler(w http.ResponseWriter, r *http.Request) {
mux.RLock()
shared2 := &SharedStruct{}
*shared2 = *shared
shared2.Status_ = new(Status)
*shared2.Status_ = *shared.Status_
mux.RUnlock()
t.ExecuteTemplate(w, "page.html", shared2)
}
如果模板只使用 shared
字段的一小部分,当然只复制这些字段就足够了。