官方推荐使用 mongo-go-driver(go.mongodb.org/mongo-driver/mongo),连接需 context 和 uri,crud 操作须注意 bson.m 与结构体解码、updateone 的 $set 用法、aggregate 游标必须 close。

Go 官方推荐且唯一维护的 MongoDB 驱动是 go.mongodb.org/mongo-driver/mongo,别用已归档的 mgo。连接前先确保 MongoDB 服务在运行(如 mongodb://localhost:27017),并安装驱动:
go get go.mongodb.org/mongo-driver/mongo
连接和插入示例:
import ( "context" "go.mongodb.org/mongo-driver/mongo" "go.mongodb.org/mongo-driver/mongo/options" "go.mongodb.org/mongo-driver/bson" ) client, err := mongo.Connect(context.TODO(), options.Client().ApplyURI("mongodb://localhost:27017")) if err != nil { panic(err) } defer client.Disconnect(context.TODO()) collection := client.Database("testdb").Collection("users") res, err := collection.InsertOne(context.TODO(), bson.M{"name": "Alice", "age": 30}) if err != nil { panic(err) } // res.InsertedID 是新文档的 ObjectID
注意:context 不可传 nil,生产环境应带超时(如 context.WithTimeout);bson.M 是 map[string]interface{} 的别名,字段名默认按字典序序列化,但不影响查询。
直接用 bson.M 接收查询结果最简单,但结构体更安全、易维护。关键点在于结构体字段必须导出(首字母大写),且用 bson tag 显式指定键名,否则无法映射:
type User struct {
ID primitive.ObjectID `bson:"_id,omitempty"` Name string `bson:"name"` Age int `bson:"age"`
}
var user User err := collection.FindOne(context.TODO(), bson.M{"name": "Alice"}).Decode(&user)
常见错误:
- 忘记加
&导致Decode报cannot decode BSON array into a *main.User - 结构体字段未导出,解码后字段值全为零值(
Name为空字符串,Age为 0) - 查询无结果时
FindOne返回mongo.ErrNoDocuments,需显式检查,不能只看err == nil
UpdateOne 第二个参数是更新操作符(如 \(set、\)inc),不是原始文档。直接传 bson.M{"name": "Bob"} 会清空整个文档,只剩这个字段:
// ❌ 错误:覆盖整个文档 collection.UpdateOne(ctx, filter, bson.M{"name": "Bob"})
// ✅ 正确:只改 name 字段 collection.UpdateOne(ctx, filter, bson.M})
// ✅ 正确:同时改多个字段 collection.UpdateOne(ctx, filter, bson.M, "\(inc": bson.M{"score": 10}})
其他要点:
- filter 必须是匹配条件(如
bson.M{"_id": id}),不能传空bson.D{}否则可能误更新第一条 - 若想插入不存在的文档,用
Upsert: true选项 -
\)
Aggregate 返回的是 *mongo.Cursor,必须调用 Next + Decode 迭代,且最后要 Close。漏掉 Close 会导致连接泄漏,尤其在高并发下:
cursor, err := collection.Aggregate(ctx, []bson.M{
{"$match": bson.M{"age": bson.M{"$gt": 25}}}, {"$group": bson.M{"_id": "$name", "count": bson.M{"$sum": 1}}},
}) if err != nil {
panic(err)
} defer cursor.Close(ctx) // ⚠️ 必须 close!
for cursor.Next(ctx)
fmt.Println(result)
} if err := cursor.Err(); err != nil { // 检查迭代过程中的错误
panic(err)
}
容易忽略的点:
-
Aggregate默认不自动分页,大数据量时加\(limit或配合SetMaxTime防止长耗时 - 管道阶段顺序敏感,比如
\)match放前面能减少后续阶段处理的数据量 - 调试聚合时,建议先在 MongoDB Shell 里验证 pipeline 是否正确,再搬进 Go 代码
真正难的不是语法,是 context 生命周期管理、错误分类处理、游标资源释放这些细节。一个没关的 cursor,可能让整个服务慢慢变慢。
golang免费学习笔记(深入):立即使用
在学习笔记中,你将探索golang的核心概念和高级技巧!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请联系我们,一经查实,本站将立刻删除。
如需转载请保留出处:https://51itzy.com/kjqy/251759.html