聚合操作主要用于处理数据并返回计算结果。聚合操作将来自多个文档的值组合在一起,按条件分组后,再进行一系列操作(如求和、平均值、最大值、最小值)以返回单个结果。

MongoDB 提供了三种执行聚合的方法:聚合管道、map-reduce 和单一目标聚合方法,本节只介绍前两种方法。

聚合管道方法

MongoDB 的聚合框架就是将文档输入处理管道,在管道内完成对文档的操作,最终将文档转换为聚合结果。

最基本的管道阶段提供过滤器,其操作类似查询和文档转换,可以修改输出文档的形式。其他管道操作提供了按特定字段对文档进行分组和排序的工具,以及用于聚合数组内容(包括文档数组)的工具。

此外,在管道阶段还可以使用运算符来执行诸如计算平均值或连接字符串之类的任务。聚合管道可以在分片集合上运行。

聚合管道方法的流程参见下图。

上图的聚合操作相当于 MySQL 中的以下语句:

select cust_id as _id, sum(amount) as total from orders where status like "%A%" group by cust_id;

MongoDB 中的聚合操作语法如下:

db.collection.aggregate([

{

$match : {< query >},

}

{

$group: {< fieldl >: < field2 >}

}

])

Query 设置统计查询条件,类似于 SQL 的 where,field1 为分类字段,要求使用 _id 名表示分类字段,field2 为包含各种统计操作符的数字型字段,如 $sum、$avg、$min 等。

这个语法看起来比较难以理解,下面给出一个示例进行对照:

db.mycol.aggregate([

{

$group : {_id : "$by_user", num_tutorial : {$sum : 1}}

}

])

相当于MySQL中的:

select by_user as _id, count(*) as num_tutorial from mycol group by by_user;

再举一个复杂的例子,按照指定条件对文档进行过滤,然后对满足条件的文档进行统计,并将统计结果输出到临时文件中。

首先插入多条文档,代码如下:

db.articles.insert([

{ "_id" : 10, "author" : "dave", "score" : 80, "views" :100 },

{ "_id" : 11, "author" : "dave", "score" : 85, "views" : 521 },

{ "_id" : 12, "author" : "ahn", "score" : 60, "views" : 1000 },

{ "_id" : 13, "author" : "li", "score" : 55, "views" : 5000 },

{ "_id" : 14, "author" : "annT", "score" : 60, "views" : 50 },

{ "_id" : 15, "author" : "1i", "score": 94, "views": 999 },

{ "_id" : 16, "author" : "ty", "score" : 95, "views": 1000 }

]);

再进行聚合分类统计,代码如下:

db.articles.aggregate([

{

$match: { $or: [{ score: { $gt: 70, $1t: 90 }}, { views: { $gte: 1000 }}]}}, { $group: { _id: null, count: { $sum: 1 }}

}

]);

最终统计结果为:

{ "_id" : null, "count" : 5 }

管道阶段的 RAM 限制为 100MB。若要允许处理大型数据集,则可使用 allowDiskUse 选项启用聚合管道阶段,将数据写入临时文件。

map-reduce 方法

MongoDB 还提供了 map-reduce 方法来执行聚合。通常 map-reduce 方法有两个阶段:首先 map 阶段将大批量的工作数据分解执行,然后 reduce 阶段再将结果合并成最终结果。

与其他聚合操作相同,map-reduce 可以指定查询条件以选择输入文档以及排序和限制结果。

map-reduce 使用自定义 JavaScript 函数来执行映射和减少操作,虽然自定义 JavaScript 与聚合管道相比提供了更大的灵活性,但通常 map-reduce 比聚合管道效率更低、更复杂。

map-reduce 可以在分片集合上运行,也可以输出到分片集合。map-reduce 的语法如下:

>db.collection.mapReduce(

function() { emit(key,value); },

function(key, values) { return reduceFunction }

{ query: document, out: collection }

)

参数说明:

function() { emit(key,value); } 为 map 映射函数,负责生成键值对序列,并作为 reduce 函数输入参数。

function(key, values) { return reduceFunction } 为 reduce 统计函数,reduce 函数的任务就是将 key-values 变成 key-value,也就是把 values 数组转换成一个单一的值 value。

query 设置筛选条件,只有满足条件的文档才会调用 map 函数。

out 为统计结果的存放集合,如果不指定则使用临时集合,但会在客户端断开后自动删除。

举例说明使用 map-Teduce 方法进行 MongoDB 文档数据的聚合。首先插入数据,数据为每位顾客 cust_id 的消费情况,代码如下:

db.order.insert([

{ "cust_id" : "l", "status" : "A", "price" : 25, "items" : [{"sku" : "mmm", "qty" : 5, "price" : 2.5 },

{ "sku" : "nnn", "qty" : 5, "price" : 2.5 }]},

{ "cust_id" : "l", "status" : "A", "price" : 25, "items" : [{"sku" : "mmm", "qty" : 5, "price" : 2.5 },

{ "sku" : "nnn", "qty" : 5, "price" : 2.5 }]},

{ "cust_id" : "2", "status" : "A", "price" : 25, "items" : [{"sku" : "mmm", "qty" : 5, "price" : 2.5 },

{ "sku" : "nnn", "qty" : 5, "price" : 2.5 }]},

{ "cust_id" : "3", "status" : "A", "price" : 25, "items" : [{"sku" : "mmm", "qty" : 5, "price" : 2.5 },

{ "sku" : "nnn", "qty" : 5, "price" : 2.5 }]},

{ "cust_id" : "3", "status" : "A", "price" : 25, "items" : [{"sku" : "mmm", "qty" : 6, "price" : 2.5 },

{ "sku" : "nnn", "qty" : 6, "price" : 2.5 }]},

])

编写 map 函数,cust_id 作为 map 的输出 key,price 作为 map 的输出 value,代码如下:

var mapFunc = function(){

emit(this.cust_id, this.price);

};

编写 reduce 函数,将相同的 map 的输出 key(cust_id) 聚合起来,这里对输出的 value 进行 sum 操作,代码如下:

var reduceFunc = function(key,values){

return Array.sum(values);

};

执行 map-reduce 任务,将 reduce 的输出结果保存在集合 map_result_result 中,代码如下:

db.order.mapReduce(mapFunc, reduceFunc, { out: { replace: 'map_result_result' }})

查看当前数据库下的所有集合,会发现新建了一个 map_result_result,此集合里保存了 map-reduce 聚合后的结果:

>show collections

map_result_result

myColl

order

>db.map_result_result.find()

{ "_id" : "l", "value" : 50.0 }

{ "_id" : "2", "value" : 25.0 }

{ "_id" : "3", "value" : 55.0 }

mongodb聚合查询优化_MongoDB聚合查询详解相关推荐

  1. mysql单表查询实例_MySQL简单查询详解-单表查询

    MySQL简单查询详解-单表查询 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.查询的执行路径 一条SQL查询语句的执行过程大致如下图所示: 1>.客户端和服务端通过my ...

  2. Django 模型层(models) 复杂查询详解

    Django 模型层(models) 复杂查询详解 一般Django orm 和原生sql混合使用 1.测试文件 只单独测试django中的某一个py文件 不一定是tests.py 1.配置 在任意一 ...

  3. 03 mysql数据查询_MySql学习day03:数据表之间的连接、查询详解

    主键: 关键字:primary key 特点:不能为null,并且唯一. 主键分类: 逻辑主键:例如ID,不代表实际的业务意义,只是用来唯一标识一条记录(推荐) 业务主键:例如username,参与实 ...

  4. mysql多表查询详解_MySQL多表查询详解上

    时光在不经意间,总是过得出奇的快.小暑已过,进入中暑,太阳更加热烈的绽放着ta的光芒,...在外面被太阳照顾的人们啊,你们都是勤劳与可爱的人啊.在房子里已各种姿势看我这篇这章的你,既然点了进来,那就由 ...

  5. php视图查询的优势,ThinkPHP视图查询详解

    搜索热词 ThinkPHP提供的视图查询应用功能十分强大,用户利用视图查询功能可以将多个数据表的字段内容按需要进行指定和筛选,组织成一个基于这些数据表的视图模型,然后就可以通过该模型直接进行多表联合查 ...

  6. oracle做子查询注意事项,Oracle子查询详解

    Oracle子查询详解,根据查询的结果(内部嵌套查询的结果)把子查询的类型分为单行子查询与多行子查询, 子查询概念 :当一个查询的结果是另一个查询的条件时,,称之为子查询. 使用子查询注意事项: 子查 ...

  7. mongodb创建集合命令db.createCollection详解

    mongodb创建集合命令db.createCollection详解 完整的命令如下: db.createCollection(name, {capped: <Boolean>, auto ...

  8. node mysql 查询_Node.js使用mysql进行查询详解

    本篇教程介绍了Node.js使用mysql进行查询详解,希望阅读本篇文章以后大家有所收获,帮助大家对Node.js的理解更加深入. < 因为返回的是个对象 var selectSql1=&quo ...

  9. Mysql 多表查询详解

    Mysql 多表查询详解 一.前言  二.示例 三.注意事项 一.前言  上篇讲到Mysql中关键字执行的顺序,只涉及了一张表:实际应用大部分情况下,查询语句都会涉及到多张表格 : 1.1 多表连接有 ...

最新文章

  1. 技术图文:如何解决 DAO 抛出的 80040154 错误?
  2. ArrayList和LinkedList的add(E)性能秘密
  3. 水质php202169,基于php的水质查询api调用代码实例
  4. MLPerf最新AI芯片跑分:谷歌TPU和英伟达打破记录
  5. try、catch、finally的执行顺序
  6. 排序算法复习—希尔排序
  7. Sqlite学习笔记(五)SQLite封锁机制
  8. 子网与子网掩码的介绍
  9. JSP字符集编码集合
  10. OpenStack Rocky Octavia 的实现与分析(零)架构简介
  11. 相信美好就能遇见美好—西安独行三日短途穷游 攻略
  12. 数学分析高等代数考研试题不断更新中
  13. 医院计算机管理工资,医院绩效管理平台:绩效工资管理系统
  14. 【MATLAB】进阶绘图 ( 双 y 轴图形 | plotyy 函数 | Histogram 统计图形 | hist 函数 )
  15. 三位数除以两位数竖式计算没有余数_三位数除两位数计算题-云簿杜同学
  16. 计算机一打开就卡在更新失败,做系统一直在正在启动画面-电脑开机后卡在“正在启动windows”界面,怎么办?...
  17. 超详细的SpringBoot+Mybatis+Vue整合笔记
  18. 基金定投是什么?定投的特点?
  19. 后台:七牛云上传图片,视频,文件
  20. 攻克微服务中的最大难点:用户数据

热门文章

  1. windows 下远程登录ubuntu服务器--realVNC
  2. wsl(windows上运行linux)安装到非C盘解决方案
  3. 传统IDC转型有了新思路 Zstack除了混合云还有托管云
  4. 高速掌握Lua 5.3 —— 字符串库 (2)
  5. FileInputStream、FileReader、FileWriter和File
  6. 新JEP将简化Java类型变异
  7. 【BZOJ 2809】 [Apio2012]dispatching
  8. 移动支付漏洞大 没密码也能买买买
  9. 中科院NLPIR中文分词java版
  10. 自己喜欢的shell终端配置