MongoDB 系列(二) C# 内嵌元素操作 聚合使用
"_id" : "639d8a50-7864-458f-9a7d-b72647a3d226",
"ParentGuid" : "00000000-0000-0000-0000-000000000000",
"Name" : "汉字",
"Describe" : "",
"Enable" : true,
"Level" : 2,
"Subject" : "语文",
"Exams" : [
{
"ExamGuid" : "6635e1c6-2a22-404e-b58c-97ed9fc49ffc",
"Grade" : "G2",
"Questions" : [
{
"_id" : "97eeeaaa-5902-4c11-bcdb-8b7a70485753"
},
{
"_id" : "628d5613-90b1-44fa-b02d-4074e86aa751"
}
]
},
{
"ExamGuid" : "b1e72692-6b63-4d98-a4bb-1b48765e7a96",
"Grade" : "G1",
"Questions" : [
{
"_id" : "628d5613-90b1-44fa-b02d-4074e86aa751"
}
]
}
]
场景1:给内嵌集合的元素添加一个新的元素
1.首先根据筛选条件确定元素
var filter = Builders<CustomKnowPoint>.Filter.And(
Builders<CustomKnowPoint>.Filter.Eq(o => o.Id, "639d8a50-7864-458f-9a7d-b72647a3d226"),
Builders<CustomKnowPoint>.Filter.Eq("Exams.ExamGuid", "6635e1c6-2a22-404e-b58c-97ed9fc49ffc"));
CustomKnowPoint为MongoDB映射的对象 Builders<CustomKnowPoint>.Filter.And 方法的参数是一个param[]数组,意思是将所有的过滤条件用And且的关系合并
var update = Builders<CustomKnowPoint>.Update.Push("Exams.$.Questions", new QuestionGuid(Guid.NewGuid().ToString()));
this.Collection.FindOneAndUpdate(filter, update);
2.执行后Questions内嵌集合就会在元素的末尾添加一条新的记录
场景2:内嵌集合的元素进行删除
过滤条件同上面的新增一样 通过对象本身的Id和内嵌元素的ExamGuid固定内嵌文档,移除文档的具体Filter对象是移除对象本身Builders<QuestionGuid>.Filter而不是Builders<CustomKnowPoint>.Filter
var update = Builders<CustomKnowPoint>.Update.PullFilter("Exams.$.Questions", Builders<QuestionGuid>.Filter.Eq("_id", "628d5613-90b1-44fa-b02d-4074e86aa751"));
this.Collection.FindOneAndUpdate(filter, update);
场景3:聚合使用
MongoDB的聚合管道将MongoDB文档在一个管道处理完毕后将结果传递给下一个管道处理。管道操作是可以重复的。表达式:处理输入文档并输出。表达式是无状态的,只能用于计算当前聚合管道的文档,不能处理其它的文档。
"_id" : "639d8a50-7864-458f-9a7d-b72647a3d226",
"ParentGuid" : "00000000-0000-0000-0000-000000000000",
"Name" : "汉字",
"Describe" : "",
"Enable" : true,
"Level" : ,
"Subject" : "语文",
"Exams" : [
{
"ExamGuid" : "6635e1c6-2a22-404e-b58c-97ed9fc49ffc",
"Grade" : "G2",
"Questions" : [
{
"_id" : "97eeeaaa-5902-4c11-bcdb-8b7a70485753"
},
{
"_id" : "628d5613-90b1-44fa-b02d-4074e86aa751"
}
]
}
]
当我们只需要返回内嵌文档部分的时候 这种场景下我们就可以用到聚合来操作了 执行MongoDB语句
db.KnowPoint.aggregate(
[
{
/*此处$unwind:将文档中的某一个数组类型字段拆分成多条,每条包含数组中的一个值。*/
$unwind : "$Exams"
},
{
/*此处$match :过滤文档筛选条件。*/
$match :
{
"Exams.ExamGuid" : "6635e1c6-2a22-404e-b58c-97ed9fc49ffc"
}
},
{
/*此处group 分组。*/
$group :
{
_id :
{
/*此处按照各种属性拼凑分组 将各属性组成一个Id的属性*/
"_id" : "$_id", "ParentGuid" : "$ParentGuid", "Name" : "$Name", "Enable" : "$Enable", "Level" : "$Level", "Subject" : "$Subject"
},
"Exams" :
{
/*此处push将之前分组的Exams再写进去*/
"$push" : "$Exams"
}
}
}
])
相应的C# 驱动程序
List<string> keys = new List<string>() { "_id", "ParentGuid", "Name", "Enable", "Level", "Subject" };
var stages = new List<IPipelineStageDefinition>();
#region 过滤条件
stages.Add(new JsonPipelineStageDefinition<BsonDocument, BsonDocument>
(
"{" +
"$unwind:\"$Exams\"" +
"}"
));
stages.Add(new JsonPipelineStageDefinition<BsonDocument, BsonDocument>
(
"{" +
"$match:" +
"{" +
"\"Exams.ExamGuid\":" +
"{" +
"$in:" + JsonConvert.SerializeObject(new List<string>() { "6635e1c6-2a22-404e-b58c-97ed9fc49ffc"}) +
"}" +
"}" +
"}"
)); stages.Add(new JsonPipelineStageDefinition<BsonDocument, BsonDocument>
(
"{" +
"$group:" +
"{" +
"_id:{ " + string.Join(",", keys.Select(o => "\"" + o + "\":\"$" + o + "\"").ToList()) + " }," +
"Exams:{$push:\"$Exams\"}" +
"}" +
"}"
));
#endregion var pipeline = new PipelineStagePipelineDefinition<BsonDocument, BsonDocument>(stages); var result = this.Context
.DbSet<BsonDocument>(nameof(KnowPoint))
.Aggregate(pipeline)
.ToList()
MongoDB 系列(二) C# 内嵌元素操作 聚合使用的更多相关文章
- MongoDB查询或修改内嵌文档
作为非关系型数据库中的佼佼者,MongoDB一大优势在于能够在一条文档中存储对象类型的数据,适当增加冗余来让数据库更好用.文档中一个对象类型的字段在MongoDB中被称为内嵌文档(Embedded) ...
- web - 块元素和内嵌元素的特征
块: 1.独占一行 2.支持所有的样式 3.不设置宽度的时候,宽度撑满整行 常用的快标签有: div,section,header,nav,footer,article,aside,ul,ol,li, ...
- MongoDB系列二(介绍).
一.特点 学习一个东西,至少首先得知道它能做什么?适合做什么?有什么优缺点吧? 传统关系型数据库,遵循三大范式.即原子性.唯一性.每列与主键直接关联性.但是后来人们慢慢发现,不要把这些数据分散到多个表 ...
- 关于selenium自动化对iframe内嵌元素的处理
今天上班闲来无聊,于是来练练自动化,结果碰上了可恶的iframe,楼主,以前也遇到过,但是一直也没搞懂怎么处理的,都是抄别人的代码,今天决定独立解决试试.首先先来认识什么是iframe,它就长下图这样 ...
- Selenium4.0+Python3系列(四) - 常见元素操作(含鼠标键盘事件)
一.写在前面 上篇文章介绍的是关于浏览器的常见操作,接下来,我们将继续分享关于元素的常见操作,建议收藏.转发! 二.元素的状态 在操作元素之前,我们需要了解元素的常见状态. 1.常见元素状态判断,傻傻 ...
- css中块级元素、内联元素(行内元素、内嵌元素)
Block element 块级元素 顾名思义就是以块显示的元素,高度宽度都是可以设置的.比如我们常用 的<div>.<p>.<ul>默认状态下都是属于块级元 ...
- MongoDB系列二
简介 MongoDB是一个基于分布式文件存储的数据库.由C++语言编写.旨在为WEB应用提供可扩展的高性能数据存储解决方案. MongoDB是一个高性能,开源,无模式的文档型数据库,是当前NoSql ...
- glance系列二:glance部署及操作
一 简单架构图示参考 更新中... 二 部署glance yum install memcached python-memcachedsystemctl enable memcached.servic ...
- MongoDB系列一(索引及C#如何操作MongoDB)
索引总概况 db.test.ensureIndex({"username":1})//创建索引 db.test.ensureIndex({"username": ...
随机推荐
- shp系列(一)——利用C++进行shp文件的读(打开)与写(创建)开言
博客背景和目的 最近在用C++写一个底层的东西,需要读取和创建shp文件.虽然接触shp文件已经几年了,但是对于shp文件内到底包含什么东西一直是一知半解.以前使用shp文件都是利用软件(如ArcGI ...
- 迭代器与函数Python学习(四)
1.1 迭代器: 迭代的工具 1.1.1 什么是迭代: 指的是一个重复的过程,每一次重复称为一次迭代,并且每一次重复的结果是下一次重复的初始值 while True: print('=====> ...
- hdu 2795 Billboard 【线段树】
给出一个高为h,宽为w的广告板,有n张广告需要贴,从第一行开始贴,尽量靠左,输出每个广告最后贴在哪一行的 先一直想不通这样建树是为什么 后来看到一篇题解里面的一句话“直到找到一个满足条件的叶子节点” ...
- ABBYY FineReader去他的光棍节,我要我的双十一
今天就是双十一,全民剁手的双十一,一年仅一次的双十一,不只是半价的双十一.....此时此刻,多少钱拿起手机在疯狂购物,又有多少人死守着电脑,不敢怠慢一丁点机会,买着买着购物车就空了,然后才发现,咦!超 ...
- Long型转换成IP段String、StringIP段转换成Long型
/** 把long类型的Ip转为一般Ip类型:xx.xx.xx.xx * * @param ip * @return */ public static String getIpFromLong(Lon ...
- 凸多边形 HRBUST - 1429 计算几何_凸包_未调完
任选一个点作为起始点,将其他点按与该点连线的极角排序,二分查询点在哪两个射线之间, 并特别判断一下边界即可. Code: #include <cstdio> #include <al ...
- CentOS 7 安装配置MySQL
环境 CentOS Linux release 7.5.1804 (Core) MySQL:mysql80-community-release-el7-1 检查: 在centos7中默认的是maria ...
- 【洛谷4941】War2 状压Dp
简单的状压DP,和NOIP2017 Day2 找宝藏 代码几乎一样.(比那个稍微简单一点) f[i][j] ,i代表点的状态,j是当前选择的点,枚举上一个选到的点k 然后从f[i-(1<< ...
- 鸟哥的linux私房菜
http://vbird.dic.ksu.edu.tw/linux_basic/linux_basic.php
- 洛谷P1004 方格取数
网络流大法吼 不想用DP的我选择了用网络流-- 建模方法: 从源点向(1,1)连一条容量为2(走两次),费用为0的边 从(n,n)向汇点连一条容量为2,费用为0的边 每个方格向右边和下边的方格连一条容 ...