我目前正在使用带有mgo lib的mongodb作为Web应用程序,但我不确定我使用它的方式是不是很好..
package db
import (
"gopkg.in/mgo.v2"
)
const (
MongoServerAddr = "192.168.0.104"
RedisServerAddr = "192.168.0.104"
)
var (
MongoSession, err = mgo.Dial(MongoServerAddr)
MDB = MongoSession.DB("message")
MCol = MDB.C("new")
MSav = MDB.C("save")
UDB = MongoSession.DB("account")
UCol = UDB.C("user")
)
我初始化db会话并创建获取集合和文档值的变量, 因此,当我需要查询集合时,我使用该变量来实现它。
就像那样:
func UserExist(username string) bool {
user := Users{}
err := db.UCol.Find(bson.M{"username": username}).One(&user)
if err != nil {
return false
} else {
return true
}
}
那么有最好的做法还是这个很好......? 感谢
答案 0 :(得分:63)
我建议不要使用这样的全局会话。相反,您可以创建一个负责所有数据库交互的类型。例如:
type DataStore struct {
session *mgo.Session
}
func (ds *DataStore) ucol() *mgo.Collection { ... }
func (ds *DataStore) UserExist(user string) bool { ... }
该设计有许多好处。一个重要的一点是它允许你同时在飞行中有多个会话,所以如果你有一个http处理程序,你可以创建一个由独立会话支持的本地会话,只针对那个请求: / p>
func (s *WebSite) dataStore() *DataStore {
return &DataStore{s.session.Copy()}
}
func (s *WebSite) HandleRequest(...) {
ds := s.dataStore()
defer ds.Close()
...
}
在这种情况下,mgo驱动程序表现良好,因为会话在内部进行缓存和重用/维护。每个会话在使用时也将由独立套接字支持,并且可能配置了独立设置,并且还将具有独立的错误处理。如果您使用单个全局会话,这些是您最终必须处理的问题。
答案 1 :(得分:2)
虽然没有直接回答你的问题,但是关于mgo会话检查你必须使用延迟/恢复,因为mgo呼叫(甚至是mgo.session.Ping)恐慌。据我所知,没有其他方法可以检查mgo会话状态(mgo godocs)。您可以使用Gustavo Niemeyer的建议,并在DataStore
类型上添加方法。
func (d *DataStore) EnsureConnected() {
defer func() {
if r := recover(); r != nil {
//Your reconnect logic here.
}
}()
//Ping panics if session is closed. (see mgo.Session.Panic())
d.Ping()
}
答案 2 :(得分:0)
使用go 1.7,在web服务器上处理mongo会话的最惯用方法是使用新的标准库包context
编写一个中间件,可以在请求上下文完成时附加defer session.Close()
() 叫做。所以你不需要记得关闭
AttachDeviceCollection = func(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
db, err := infra.Cloner()
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
collection, err := NewDeviceCollection(db)
if err != nil {
db.Session.Close()
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
ctx := context.WithValue(r.Context(), DeviceRepoKey, collection)
go func() {
select {
case <-ctx.Done():
collection.Session.Close()
}
}()
next.ServeHTTP(w, r.WithContext(ctx))
})
}