怎么重用模型类型来查询和解组“混合”结果(Mgo聚合)
学习Golang要努力,但是不要急!今天的这篇文章《怎么重用模型类型来查询和解组“混合”结果(Mgo聚合)》将会介绍到golang等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!
问题内容
假设我们有 2 个集合:"users"和"posts",由以下类型建模:
type User struct {
ID string `bson:"_id"`
Name string `bson:"name"`
Registered time.Time `bson:"registered"`
}
type Post struct {
ID string `bson:"_id"`
UserID string `bson:"userID"`
Content string `bson:"content"`
Date time.Time `bson:"date"`
}这些可以在存储/检索单个文档甚至文档集合时使用,例如:
usersColl := sess.DB("").C("users")
postsColl := sess.DB("").C("posts")
// Insert new user:
u := &User{
ID: "1",
Name: "Bob",
Registered: time.Now(),
},
err := usersColl.Insert(u)
// Handle err
// Get Posts in the last 10 mintes:
var posts []*Post
err := postsColl.Find(
bson.M{"date": bson.M{"$gt": time.Now().Add(-10 * time.Minute)}},
).Limit(20).All(&posts)
// Handle err如果我们使用聚合来获取这些文档的混合怎么办?例如Collection.Pipe():
// Query users with their posts:
pipe := collUsers.Pipe([]bson.M{
{
"$lookup": bson.M{
"from": "posts",
"localField": "_id",
"foreignField": "userID",
"as": "posts",
},
},
})
var doc bson.M
it := pipe.Iter()
for it.Next(&doc) {
fmt.Println(doc)
}
// Handle it.Err()我们在单个查询中查询用户的帖子。结果是用户和帖子的混合。我们如何重用我们的User和Post模型类型,而不必将结果作为“原始”文档(类型bson.M)处理?
正确答案
上面的查询返回“几乎”匹配User文档的文档,但它们也包含每个用户的帖子。所以基本上结果是一系列 嵌入User了Post数组或切片的文档。
__
一种方法是向自身添加一个Posts []*Post字段User,我们将完成:
type User struct {
ID string `bson:"_id"`
Name string `bson:"name"`
Registered time.Time `bson:"registered"`
Posts []*Post `bson:"posts,omitempty"`
}User虽然这可行,但Posts仅仅为了单个查询而扩展似乎“过大”
。如果我们继续沿着这条路走下去,我们的User类型会变得臃肿,包含许多用于不同查询的“额外”字段。更不用说如果我们填写Posts字段并保存用户,这些帖子最终会保存在User文档中。不是我们想要的。
另一种方法是创建一个UserWithPosts类型 copyingUser并添加一个Posts []*Post字段。不用说这是丑陋和不灵活的(任何更改User都必须UserWithPosts手动反映)。
使用结构嵌入
我们可以利用结构嵌入(重用现有的和类型)
,而不是修改原来的,而不是从“零开始”User创建新类型:UserWithPostsUser``Post
type UserWithPosts struct {
User `bson:",inline"`
Posts []*Post `bson:"posts"`
}注意 bson[标记值](https://stackoverflow.com/questions/10858787/what-are-the-uses-
for-tags-in-go/30889373#30889373)",inline"。这记录在bson.Marshal()andbson.Unmarshal()(我们将使用它进行解组):
inline Inline the field, which must be a struct or a map. Inlined structs are handled as if its fields were part of the outer struct. An inlined map causes keys that do not match any other struct field to be inserted in the map rather than being discarded as usual.
通过使用嵌入和",inline"标签值,UserWithPosts类型本身将成为解组User文档的有效目标,其Post []*Post字段将成为查找的完美选择"posts"。
使用它:
var uwp *UserWithPosts
it := pipe.Iter()
for it.Next(&uwp) {
// Use uwp:
fmt.Println(uwp)
}
// Handle it.Err()或一步获得所有结果:
var uwps []*UserWithPosts err := pipe.All(&uwps) // Handle error
的类型声明UserWithPosts可能是也可能不是本地声明。如果您在其他地方不需要它,它可以是您执行和处理聚合查询的函数中的本地声明,因此它不会使您现有的类型和声明膨胀。如果您想重用它,您可以在包级别声明它(导出或未导出),并在需要的地方使用它。
修改聚合
另一种选择是使用
MongoDB$replaceRoot来“重新排列”结果文档,因此“简单”结构将完美地覆盖文档:
// Query users with their posts:
pipe := collUsers.Pipe([]bson.M{
{
"$lookup": bson.M{
"from": "posts",
"localField": "_id",
"foreignField": "userID",
"as": "posts",
},
},
{
"$replaceRoot": bson.M{
"newRoot": bson.M{
"user": "$$ROOT",
"posts": "$posts",
},
},
},
})通过这种重新映射,结果文档可以像这样建模:
type UserWithPosts struct {
User *User `bson:"user"`
Posts []*Post `bson:"posts"`
}请注意,虽然这有效,但posts所有文档的字段将从服务器获取两次:一次作为posts返回文档的字段,一次作为 ;
的字段user。我们不映射/使用它,但它存在于结果文档中。因此,如果选择此解决方案,user.posts则应删除该字段,例如使用一个$project阶段:
pipe := collUsers.Pipe([]bson.M{
{
"$lookup": bson.M{
"from": "posts",
"localField": "_id",
"foreignField": "userID",
"as": "posts",
},
},
{
"$replaceRoot": bson.M{
"newRoot": bson.M{
"user": "$$ROOT",
"posts": "$posts",
},
},
},
{"$project": bson.M{"user.posts": 0}},
})终于介绍完啦!小伙伴们,这篇关于《怎么重用模型类型来查询和解组“混合”结果(Mgo聚合)》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!
聊聊有关C中用Go的限制
- 上一篇
- 聊聊有关C中用Go的限制
- 下一篇
- 为何用goinstall后导入web.go报错了
-
- Golang · Go问答 | 1年前 |
- 在读取缓冲通道中的内容之前退出
- 139浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 戈兰岛的全球 GOPRIVATE 设置
- 204浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将结构作为参数传递给 xml-rpc
- 325浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何用golang获得小数点以下两位长度?
- 478浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何通过 client-go 和 golang 检索 Kubernetes 指标
- 486浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将多个“参数”映射到单个可变参数的习惯用法
- 439浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将 HTTP 响应正文写入文件后出现 EOF 错误
- 357浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 结构中映射的匿名列表的“复合文字中缺少类型”
- 352浏览 收藏
-
- Golang · Go问答 | 1年前 |
- NATS Jetstream 的性能
- 101浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将复杂的字符串输入转换为mapstring?
- 440浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 相当于GoLang中Java将Object作为方法参数传递
- 212浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何确保所有 goroutine 在没有 time.Sleep 的情况下终止?
- 143浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3180次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3391次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3420次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4526次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3800次使用
-
- 老师代码没有自动跟踪?
- 2023-03-07 439浏览
-
- c程序fork并等待golang进程状态
- 2023-03-05 262浏览
-
- GOLANG使用Context管理关联goroutine的方法
- 2022-12-28 193浏览
-
- 怎么用Golang将MySQL表转储为JSON
- 2023-03-07 188浏览
-
- Golang 检查字符串是否为有效路径?
- 2023-03-10 500浏览

