mongodb联表查询_mongodb聚合查询

mongodb联表查询_mongodb聚合查询  在使用MongoDB存储数据的时候,我们查询的时候,有时候难免会需要进行连表查询。但是MongoDB本身是非关系性数据库,连表查询,很多时候,需要我们自己在代码里手工操作。但是从MongoDB3.2版本过后,我们可以使用lookup∗∗∗进行连表查询。下面就简单介绍一下MongoDB的∗∗∗lookup∗∗∗进行连表查询。下面就简单介绍一下MongoDB的∗∗∗lookup***进行…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

  在使用MongoDB存储数据的时候,我们查询的时候,有时候难免会需要进行连表查询。但是MongoDB本身是非关系性数据库,连表查询,很多时候,需要我们自己在代码里手工操作。但是从 MongoDB 3.2 版本过后,我们可以使用 $lookup 进行连表查询。下面就简单介绍一下 MongoDB 的 $lookup 的简单使用。

  比如现在我们有两张表, userorder 表。其中 user 表中的字段有 _iduidnameageorder 表中的字段有:_iduidproductmoney; 两张表存储的数据为:

users = [{
  _id: ObjectId("5af2b2c6b138c267e414c072"), uid: "uid000", name: "小红", age: 26 }, { _id: ObjectId("5af2b2c6b138c267e414c073"), uid: "uid001", name: "小芳", age: 27 }]
orders = [{
  _id: ObjectId("4af2b2c6b138c267e414c071"), uid: "uid000", product: "产品1", money: 100 }, { _id: ObjectId("4af2b2c6b138c267e414c072"), uid: "uid000", product: "产品2", money: 200 }, { _id: ObjectId("4af2b2c6b138c267e414c073"), uid: "uid001", product: "产品1", money: 100 }, { _id: ObjectId("4af2b2c6b138c267e414c074"), uid: "uid001", product: "产品2", money: 200 }]

假如现在有两个需求:

  1. 查询用户信息并且显示该用户的总消费金额(用户名、年龄、总消费金额)
  2. 查询用户的订单信息(订单id、产品、价格、用户名)

1. 首先来看第一个需求:

 这个需求如果我们不考虑连表,只考虑关联的话,应该是

  1. 先查询出用户表所有的数据
  2. 在订单表中求出每一个用户的消费总金额
  3. 遍历用户和订单数据,然后一一通过 uid 进行匹配对应。

 如果按照我们的数据库连表来说:那应该是我们查询 user 表关联到 order 表,然后分组根据 uid 统计求和;下面来看一看具体的实现方式。

1.1 连表查询
db.user.aggregate([{
  $lookup: { // 左连接
    from: "order", // 关联到order表
    localField: "uid", // user 表关联的字段
    foreignField: "uid", // order 表关联的字段
    as: "orders"
  }
}]);

这个时候出来的结果应该为:

users = [{
  _id: ObjectId("5af2b2c6b138c267e414c072"), uid: "uid000", name: "小红", age: 26, orders: [{ _id: ObjectId("4af2b2c6b138c267e414c071"), uid: "uid000", product: "产品1", money: 100 }, { _id: ObjectId("4af2b2c6b138c267e414c072"), uid: "uid000", product: "产品2", money: 200 }] }, { _id: ObjectId("5af2b2c6b138c267e414c073"), uid: "uid001", name: "小芳", age: 27, orders: [{ _id: ObjectId("4af2b2c6b138c267e414c073"), uid: "uid001", product: "产品1", money: 100 }, { _id: ObjectId("4af2b2c6b138c267e414c073"), uid: "uid001", product: "产品1", money: 200 }] }]
1.2 拆分 orders 数组
{
  $unwind: { // 拆分子数组
    path: "$orders",
    preserveNullAndEmptyArrays: true // 空的数组也拆分
  }
}

这个时候的数据结果应该是这样的

[{
  _id: ObjectId("5af2b2c6b138c267e414c072"),
  uid: "uid000",
  name: "小红",
  age: 26,
  orders: {
    _id: ObjectId("4af2b2c6b138c267e414c071"),
    uid: "uid000",
    product: "产品1",
    money: 100
  }
}, {
  _id: ObjectId("5af2b2c6b138c267e414c072"),
  uid: "uid000",
  name: "小红",
  age: 26,
  orders: {
    _id: ObjectId("4af2b2c6b138c267e414c072"),
    uid: "uid000",
    product: "产品2",
    money: 200
  }
} …… ]
1.3 分组求和并返回字段数据
{
  $group: { // 分组查询
    _id: "$_id",
    name: { $first: "$name" },
    age: { $first: "$age" },
    money: {$sum: "$orders.money"}
  }
}

这样就查询出了我们所需要的数据。将代码总结一下为:

db.user.aggregate([{
  $lookup: { // 左连接
    from: "order", // 关联到order表
    localField: "uid", // user 表关联的字段
    foreignField: "uid", // order 表关联的字段
    as: "orders"
  }
}, {
  $unwind: { // 拆分子数组
    path: "$orders",
    preserveNullAndEmptyArrays: true // 空的数组也拆分
  }
}, { // 分组求和并返回
  $group: { // 分组查询
    _id: "$_id",
    name: { $first: "$name" },
    age: { $first: "$age" },
    money: {$sum: "$orders.money"}
  }
}]);

2. 查询用户的订单信息

2.1 连表查询

这个时候的连表是 order 表 跟 user 表关联(上一个是 user 表 和 order 表关联)

{
  $lookup: {
    from: "users",
    localField: "openid",
    foreignField: "openid",
    as: "u"
  }
}
2.2 拆分子数组
{ $unwind: "$u" }
2.3 只返回需要的字段

将 user 中需要返回的字段,提到子目录来

{$addFields: {  name: "$u.name" }}
2.4 返回最终需要的字段结果
{ 
  $project: {
    _id: 1,
    product: 1,
    money: 1,
    name: 1
  }
}

最终的代码为:

db.order.aggregate([{
  $lookup: {
    from: "users",
    localField: "openid",
    foreignField: "openid",
    as: "u"
  }
}, {
  $unwind: "$u"
}, {
  $addFields: {  name: "$u.name" }
}, {
  $project: {
    _id: 1,
    product: 1,
    money: 1,
    name: 1
  }
}]);

虽然在 MongoDB 3.2 后我们能够进行连表查询了,方便了很多。但是其实 MongoDB
本身是非关系性数据库。如果需要进行频繁的这种连表查询,我们可以考虑优化我们的数据库表。比如在订单表里面,每一条的订单记录都把我们的用户信息放进去。

[{
  _id: ObjectId("4af2b2c6b138c267e414c071"),
  uid: "uid000",
  product: "产品1",
  money: 100,
  user: {
    _id: ObjectId("5af2b2c6b138c267e414c072"),
    uid: "uid000",
    name: "小红",
    age: 26
  }
}, {
  _id: ObjectId("4af2b2c6b138c267e414c071"),
  uid: "uid000",
  product: "产品1",
  money: 100,
  user: {
    _id: ObjectId("5af2b2c6b138c267e414c072"),
    uid: "uid000",
    name: "小红",
    age: 26
  }
}]

这个时候,在实现两个需求就很简单了:

// 1. 查询用户信息并且显示该用户的总消费金额(用户名、年龄、总消费金额)
db.order.aggregate([{ // 根据 uid 求和
  $group: {
    _id: '$user.uid',
    money: {
      $sum: "$money"
    },
    name: { $first: "$user.name" },
    age: { $first: "$user.age" }
  }
}]);

// 2. 查询用户的订单信息(订单id、产品、价格、用户名)
db.order.aggregate([{
  {$addFields: {  name: "$user.name" }}
}, { // 根据 uid 求和
  $project: {
    _id: 1,
    money: 1,
    product: 1,
    name: 1
  }
}]);
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/194012.html原文链接:https://javaforall.net

(0)
上一篇 2025年11月14日 上午10:22
下一篇 2025年11月14日 上午11:01


相关推荐

  • php arcsin函数,excel如何计算反三角函数

    excel如何计算反三角函数Excel中计算反三角函数需要用到反余弦函数(ACOS)、反正弦函数(ASIN)和反正切函数(ATAN)。函数ACOS是用来计算指定数值的反余弦值的,公式为:=ACOS(number)。函数ASIN是用来计算指定数值的反正弦值的,公式为:=ASIN(number)。函数ATAN是用来计算指定数值的反正切值的,公式为:=ATAN(number)。反余弦函数的使用1、反余弦…

    2022年4月8日
    262
  • mysql查询前10条记录「建议收藏」

    mysql查询前10条记录「建议收藏」https://blog.csdn.net/xc_gxf/article/details/8149042

    2025年10月9日
    4
  • 常用的安全渗透测试工具(渗透测试工具)

    应用程序安全性并不新鲜,但它在需求、复杂性和深度方面正迅速增长。随着网络犯罪自疫情爆发以来增长了近600%,越来越多的SaaS企业开始争相保护他们的应用程序。即使那些运行最新端点保护的系统也面临重大漏洞。然而随之而来的一个问题是:即便采取了这些安全防护措施,能保护自己不会受到网络攻击吗?答案在于应用程序安全测试解决方案,该解决方案可以主动测试您的代码是否存在错误、关键漏洞和需要全面改进的领域。接下来是常见几种测试。 渗透测试 部署自动化工具,如SAST、DAST、RAST和IAST

    2022年4月16日
    64
  • setAttribute改变属性,动态改变类

    setAttribute改变属性,动态改变类

    2022年2月22日
    46
  • web前端——p标签

    web前端——p标签lt p gt 标签 Paragraph 的首字母 段落的意思 用 lt p gt lt p gt 标签包裹的内容默认是占一行 相当于打了一个回车

    2026年3月18日
    1
  • 适合于初学者—软路由全探索系列(一):探索 VMware 虚拟机旁路由安装及设置

    适合于初学者—软路由全探索系列(一):探索 VMware 虚拟机旁路由安装及设置软路由系列:VMware虚拟机旁路由配置全过程一、背景知识1、路由器2、软路由3、旁路由二、旁路由的网络配置及特性1、网络配置2、网络特性三、OpenWrt旁路由虚拟机安装1、安装准备一、背景知识1、路由器路由器(Router)是连接两个或多个网络的硬件设备,在网络间起网关的作用,是读取每一个数据包中的地址然后决定如何传送的专用智能性的网络设备。它能够理解不同的协议,例如某个局域网使用的以太网协议,因特网使用的TCP/IP协议。这样,路由器可以分析各种不同类型网络传来的数据包的

    2022年5月22日
    176

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号