MongoDB - 文档之间的关系 + _sort和投影
1. 文档对象之间的关系
- 一对一 (one TO one)
- 例如: 夫妻 (一个丈夫 对应 一个妻子)
- 在MongoDB中, 可以通过内嵌文档的形式来体现出一对一的关系
演示:
首先在 my_test
数据库中创建一个集合, wifeAndHusband
, 并向集合中插入数据
{
name:"黄蓉",
husband:{
name:"郭靖"
}
> db.wifeAndHusband.find().pretty()
{
"_id" : ObjectId("6118eb8a47935623a0ff6300"),
"name" : "黄蓉",
"husband" : {
"name" : "郭靖"
}
}
一个文档对象一旦被嵌入到另一个文档对象中就绝不可能再被嵌入到其他文档对象中,因此可以体现出一对一的关系
- 一对多 (one TO many) / 多对一 (many TO one) 例如 :
- 父母 To 孩子
- 用户 To 订单
- 文章 To 评论
- 也可以通过内嵌文档来映射一对多的关系, 拿文章和评论举例, 一个文章可以包含多个评论, 只需要在文章中添加一个评论属性, 属性中包含的是一个数组, 数组中为多个评论对象
**问题**:
评论对于文章来说是可以无限多的, 但当评论中的数据多到一定程度后, 会造成单一数据库内存过大的问题,不利于维护
具体实现:
使用 用户 to 订单
距离
- 创建一个集合
user
, 插入以下数据
> db.user.insertMany([{name:'刘俊熙'},{name:'龙猫不热'}])
> db.user.find()
- 创建订单集合
order
,插入数据
db.order.insertOne(
{
// 用户 刘俊熙对应的订单
list:['苹果','香蕉','西瓜']
}
)
但是这样还不能体现该订单是对应 刘俊熙
数据的, 如何处理?
- 将
刘俊熙
数据的_id
的值一起添加到order
集合中, 作为标识
db.order.insertOne(
{
list:['苹果','香蕉','西瓜'],
user_id:ObjectId("6118f452928b3b5423acb15f")
}
)
db.order.find()
通过唯一性的_id
作为参照插入到另一个文档对象中, 由此实现 一对多的关系
查询一对多
要求: 查询刘俊熙
的订单
db.user.findOne({name:"刘俊熙"})._id
db.order.find({user_id:db.user.findOne({name:"刘俊熙"})._id})
- 多对多 (manyTO many)
- 分类 - 商品
- 老师 - 学生
例如: 在teachers
集合中插入以下数据
db.teachers.insert([
{name:"洪七公"},
{name:"黄药师"},
{name:"龟仙人"}
]);
db.teachers.find()
那么,要怎么体现出多对多
的关系呢?
db.stus.insert([
// 如果, 郭靖既是洪七公的徒弟又是黄药师的徒弟,要怎么在数据库中体现呢?
{name:"郭靖"}
])
体现
db.stus.insert([
{
name:"郭靖",
tech_id:[
// 添加洪七公的 _id 作为tech_id 的其中一条数据
ObjectId("61192a9b928b3b5423acb166"),
// 再把洪七公的 _id 也添加进来
ObjectId("61192a9b928b3b5423acb167")
]
}
])
运行
这时候就能体现出 多对多
的关系了
2. _sort 和 投影
2.1 _sort 排序
在emp
集合中执行db.emp.find()
可以看出, 默认情况下, 查询出来的数据是根据_id
由小到大升序排列
如果我们想要顺序按照工资sal
排序要怎么办呢?
- 在 MongoDB 中使用
sort()
方法对数据进行排序,sort() 方法可以通过参数指定排序的字段,并使用1
和-1
来指定排序的方式,其中1 为升序排列
,而-1 是用于降序排列
语法:
>db.COLLECTION_NAME.find().sort({KEY:1})
代码改变如下:
db.emp.find().sort({sal:1})
这时候就可以实现sal
升序排列的效果了
那么这时候如果我们把代码改成下面这样, 运行结果又会是什么样的呢?
db.emp.find().sort({sal:1,empno:-1})
执行一下,我们来看效果
解释:
- 如果在
sort()
方法中传入多个排序条件, 会先按照最前面的条件进行升序或降序, 如果两条数据的值一样, 则会再运行第二个排序条件 - 拿上面的代码举例, 在排序的时候, 会先按照
sal
工资数据升序排列
, 如果有两个员工的sal
工资相同, 那么就会执行第二个排序条件, 按照empno
部门编号降序排列
2.2 投影
在MongoDB中,投影表示仅选择所需要字段的数据,而不是选择整个文档字段的数据。如果某个文档有5
个字段,但只要显示3
个字段,那么就只选择3
个字段吧,这样做是非常有好处的。
MongoDB的find()
方法,在 MongoDB 查询文档中此方法接收的第二个可选参数是要检索的字段列表
。 在MongoDB中,当执行find()方法时,它默认将显示文档的所有字段
。为了限制显示的字段,需要将字段列表对应的值设置为1
或0
。1用于显示字段
,而0用于隐藏字段
。
具有投影的find()
方法的基本语法如下:
>db.COLLECTION_NAME.find({},{KEY:1})
在emp
集合中有以下数据
要求: 在查询的时候只显示 ename
和 job
, 其他字段不显示, 代码如下:
> db.emp.find({},{ename:1,job:1})
这时候如果我们连 _id
字段也不想显示要怎么做?
**注意**,在执行find()方法时,始终都会显示_id字段,如果不想要此字段,则需要将其设置为0
。
> db.emp.find({},{ename:1,job:1,_id:0})
MongoDB - 文档之间的关系 + _sort和投影的更多相关文章
- mongoDB 文档概念
mongoDB 文档概念 什么是文档 文档是 mongodb 基本的数据组织单元,类似于mysql 中的记录 文档由多个键值对组成,每个键值对表达一个数据项 属于 bson 数据 ps: bson ...
- mongoDB 文档操作_删
mongoDB 文档删除 MySQL对比 mysql delete from table where ... mongo db.collection.deleteOne(query) 删除函数 del ...
- MongoDB文档的基本操作
1. MongoDB的安装方法 (1)下载MongoDB 相应的版本: (2)设置数据文件和日志文件的存放目录: (3)启动MongoDB服务: (4)将MongoDB作为服务启动. 2. Mongo ...
- mongodb文档支持的数据类型
版权声明:转载请标明来源. https://blog.csdn.net/u014285882/article/details/25510377 1. 存储类型 mongodb文档相似于json,但不是 ...
- MongoDB 文档的查询和插入操作
MongoDB是文档型数据库,有一些专门的术语,和关系型DB相似,但也有差异,例如,Collection类似于关系型DB的Table,document类似于row,key/value pair类似于c ...
- MongoDB文档、集合、数据库简介
文档 概述 文档是MongoDB的核心概念,是数据的基本单元,非常类似于关系数据库中的行.在MongoDB中,文档表示为键值对的一个有序集.MongoDB使用Javascript shell,文档的表 ...
- mongoDB 文档操作_查
基本查询命令 find 查找复合条件的所有文档 命令 db.collection.find(query,field) 参数 query 查找条件 格式: {ssss:"xxx"}是 ...
- 读《深入理解Elasticsearch》点滴-对象类型、嵌套文档、父子关系
一.对象类型 1.mapping定义文件 "title":{ "type":"text" }, "edition":{ ...
- MongoDB文档(二)--查询
(一)查询文档 查询文档可以使用以下方法 # 以非结构化的方式显示所有的文档 db.<collectionName>.find(document) # 以结构化的方式显示所有文档 db.& ...
随机推荐
- springboot-6-springSecurity
一.Spring Security的基本配置 安全需要在设计网站之初就需要做好设计 可以做到: 功能权限 访问权限 菜单权限 这些权限虽然用拦截器过滤器也能实现,但是很麻烦,所以我们一般使用框架实现 ...
- 三分钟入门 InnoDB 存储引擎中的表锁和行锁
各位对 "锁" 这个概念应该都不是很陌生吧,Java 语言中就提供了两种锁:内置的 synchronized 锁和 Lock 接口,使用锁的目的就是管理对共享资源的并发访问,保证数 ...
- MYSQL一个设备上的主从复制实现-windows
只记录一次在一个设备上实现mysql主从复制的过程,很详细,建议收藏,用到的时候照着步骤做就可以,会记录所有需要注意到的细节和一些容易遇到的坑以及解决办法! 如果需要在同一台电脑(服务器)上实现mys ...
- Linux基础服务搭建综合
Linux服务综合搭建的文章目录 =============================================== 1.foundation创建yum仓库 2.部署DNS 3.将YUM源 ...
- 第二章 Vant的v-cell的v-for使用
遍历加载list数据,类似el-table 一.问题描述 由于VantUI中没有类似于el-table的table组件,看官网的范例,只找到了cell组件有v-for
- JMeter Gui – TestElement约定[转]
转自https://www.cnblogs.com/yigui/p/7615635.html 在编写任何JMeter组件时,必须注意某些特定的约定--如果JMeter环境中正确地运行JMeter组件, ...
- synchronized 加锁 this 和 class 的区别!
synchronized 是 Java 语言中处理并发问题的一种常用手段,它也被我们亲切的称之为"Java 内置锁",由此可见其地位之高.然而 synchronized 却有着多种 ...
- 什么是RSA
一.RSA引入: RSA是什么,嗯,这是一个好问题,有没有兴趣啊 二.RSA的解释: RSA是一种加密方式,它是现代密码学的代表(什么是现代密码学,这个吗,我感觉就是我们所使用的密码的加密的方式之一可 ...
- Nginx 文件名逻辑漏洞(CVE-2013-4547)
影响版本 Nginx 0.8.41 ~ 1.4.3 / 1.5.0 ~ 1.5.7 漏洞成因 这个漏洞其实和代码执行没有太大关系,其主要原因是错误地解析了请求的URI,错误地获取到用户请求的文件名,导 ...
- 五、从GitHub浏览Prism示例代码的方式入门WPF下的Prism之MVVM中的EventAggregator
这一篇我们主要再看完示例12.13后,写了个例子,用于再Modules下执行ApplicationCommands,使用IActiveAware执行当前View的Commands,或者Applicat ...