MongoDB 聚合Group(一)
原文:http://blog.csdn.net/congcong68/article/details/45012717
一.简介
db.collection.group()使用JavaScript,它受到了一些性能上的限制。大多数情况下,$ group在Aggregation Pipeline提供了一种具有较少的限制适用的替代。可以通过指定的键的集合中的文档和执行简单的聚合函数。在2.2版本中,返回的数组可以包含最多20000个元素;即最多20000个独特的分组。
我们比较熟悉的group by 的sql语句select key from table groupby key,而mongoDB没提供SQL那样通过Group By就轻松实现数据库的分组功能,我们通过接口来实现的
db.collection.group({ key, reduce, initial[, keyf] [, cond] [, finalize] })
|
key |
作为分组的key |
|
reduce |
一个聚合函数操作文档的分组操作期间。这些函数可以返回一个sum或count。该函数接受两个参数:当前文档和这个群体聚集的结果文档。 |
|
initial |
初始化聚合结果文档变量,为空时自动为每列提供初始变量。 |
|
keyf |
可选。替代的key 字段。指定一个函数创建一个“key object”作为分组的key。使用keyf而是通过group by领域而不是现有的文档域键组。 |
|
cond |
过滤条件 |
|
finalize |
在db.collection.group()返回最终结果之前,此功能可以修改的结果文档或替换的结果文档作为一个整体。 |
二.Mongo VUE操作Group By
1.MonogoDB数据库中添加了订单的数据
- /* 0 */
- {
- "_id" : ObjectId("552a330e05c27486b9b9b650"),
- "_class" : "com.mongo.model.Orders",
- "onumber" : "002",
- "date" : ISODate("2014-01-03T16:03:00Z"),
- "cname" : "zcy",
- "item" : {
- "quantity" : 1,
- "price" : 4.0,
- "pnumber" : "p002"
- }
- }
- /* 1 */
- {
- "_id" : ObjectId("552a331d05c275d8590a550d"),
- "_class" : "com.mongo.model.Orders",
- "onumber" : "003",
- "date" : ISODate("2014-01-04T16:03:00Z"),
- "cname" : "zcy",
- "item" : {
- "quantity" : 10,
- "price" : 2.0,
- "pnumber" : "p001"
- }
- }
- /* 2 */
- {
- "_id" : ObjectId("552a333105c2f28194045a72"),
- "_class" : "com.mongo.model.Orders",
- "onumber" : "003",
- "date" : ISODate("2014-01-04T16:03:00Z"),
- "cname" : "zcy",
- "item" : {
- "quantity" : 30,
- "price" : 4.0,
- "pnumber" : "p002"
- }
- }
- /* 3 */
- {
- "_id" : ObjectId("552a333f05c2b62c01cff50e"),
- "_class" : "com.mongo.model.Orders",
- "onumber" : "004",
- "date" : ISODate("2014-01-05T16:03:00Z"),
- "cname" : "zcy",
- "item" : {
- "quantity" : 5,
- "price" : 4.0,
- "pnumber" : "p002"
- }
- }
2.MongoDB实现分组并统计
1)我们要对日期和产品编码进行分组,并计算相同的产品的数量
Sql语句:Select date, pnumber,sum(quantity) as total from orders,items group by date, pnumber(少了两张表的关联的条件)
MongoDB:
db.orders.group({
key: { date:1,'item.pnumber':1},
initial : {"total":0},
reduce : function Reduce(doc, out) {
out.total+=doc.item.quantity
} });
结果:
2)实现一天卖出了多少个产品,金额是多少,平均价格是多少
db.orders.group({
key: {date:1},
initial :{"total":0,"money":0},
reduce : function Reduce(doc, out) {
out.total+=doc.item.quantity;
out.money+=doc.item.quantity*doc.item.price;
},
finalize : function Finalize(out) {
out.avg=out.money/out.total
returnout;
}
});
结果:
3)keyf的使用
keyf 对日期进行处理并以作为key来进来分组
db.orders.group({
keyf: function (doc){
return{'month':doc.date.getMonth()+1};
},
initial :{"total":0,"money":0},
reduce : function Reduce(doc, out) {
out.total+=doc.item.quantity;
out.money+=doc.item.quantity*doc.item.price;
},
finalize : function Finalize(out) {
out.avg=out.money/out.total
returnout;
}
});
结果:
三.Java MongoDB 实现
1)我们要对日期和产品编码进行分组,并计算相同的产品的数量
- <strong> </strong> @Override
- public void getGroupCount(String collectionName) {
- BasicDBObject key = new BasicDBObject();
- key.put("date", 1);
- key.put("item.pnumber", 1);
- //条件
- BasicDBObject cond = new BasicDBObject();
- //初始化
- BasicDBObject initial = new BasicDBObject();
- initial.append("total", 0);
- //reduce
- String reduce = "function Reduce(doc, out) { " +
- " out.total+=doc.item.quantity;" +
- "}";
- SimpleDateFormat format=new SimpleDateFormat("yyyy-mm-dd");
- BasicDBList groupList=(BasicDBList) mongoTemplate.getCollection(collectionName).group(key, cond, initial, reduce);
- if(groupList!=null&&groupList.size()>0){
- System.out.println("date item.pnumber total");
- for(int i=0;i<groupList.size();i++){
- BasicDBObject obj=(BasicDBObject) groupList.get(i);
- System.out.println(format.format(obj.getDate("date"))+" "+obj.getString("item.pnumber")+" "+obj.getInt("total"));
- }
- }
- }
结果:
2)实现一天卖出了多少个产品,金额是多少,平均价格是多少
- @Override
- public void getGroupAvg(String collectionName) {
- BasicDBObject key = new BasicDBObject();
- key.put("date", 1);
- //条件
- BasicDBObject cond = new BasicDBObject();
- //初始化
- BasicDBObject initial = new BasicDBObject();
- initial.append("total", 0);
- initial.append("money", 0.0);
- //reduce
- String reduce = "function Reduce(doc, out) { " +
- " out.total+=doc.item.quantity;" +
- " out.money+=doc.item.quantity*doc.item.price;" +
- "}";
- String finalize="function Finalize (out) { " +
- " out.avg=out.money/out.total;" +
- " return out;" +
- "}";
- SimpleDateFormat format=new SimpleDateFormat("yyyy-mm-dd");
- BasicDBList groupList=(BasicDBList) mongoTemplate.getCollection(collectionName).group(key, cond, initial, reduce, finalize);
- if(groupList!=null&&groupList.size()>0){
- System.out.println("date total money avg");
- for(int i=0;i<groupList.size();i++){
- BasicDBObject obj=(BasicDBObject) groupList.get(i);
- System.out.println(format.format(obj.getDate("date"))+" "+obj.getInt("total")+" "+obj.getInt("money")+" "+obj.getDouble("avg"));
- }
- }
- }
结果:
MongoDB 聚合Group(一)的更多相关文章
- mongodb MongoDB 聚合 group
MongoDB 聚合 MongoDB中聚合(aggregate)主要用于处理数据(诸如统计平均值,求和等),并返回计算后的数据结果.有点类似sql语句中的 count(*). 基本语法为:db.col ...
- mongodb MongoDB 聚合 group(转)
MongoDB 聚合 MongoDB中聚合(aggregate)主要用于处理数据(诸如统计平均值,求和等),并返回计算后的数据结果.有点类似sql语句中的 count(*). 基本语法为:db.col ...
- mongodb聚合 group
MongoDB中聚合(aggregate)主要用于处理数据(诸如统计平均值,求和等),并返回计算后的数据结果.有点类似sql语句中的 count(*). 基本语法为:db.collection.agg ...
- MongoDB 聚合管道(Aggregation Pipeline)
管道概念 POSIX多线程的使用方式中, 有一种很重要的方式-----流水线(亦称为"管道")方式,"数据元素"流串行地被一组线程按顺序执行.它的使用架构可参考 ...
- Mongodb学习笔记四(Mongodb聚合函数)
第四章 Mongodb聚合函数 插入 测试数据 ;j<;j++){ for(var i=1;i<3;i++){ var person={ Name:"jack"+i, ...
- 浅析mongodb中group分组
这篇文章主要介绍了浅析mongodb中group分组的实现方法及示例,非常的简单实用,有需要的小伙伴可以参考下. group做的聚合有些复杂.先选定分组所依据的键,此后MongoDB就会将集合依据选定 ...
- MongoDB 聚合
聚合操作过程中的数据记录和计算结果返回.聚合操作分组值从多个文档,并可以执行各种操作,分组数据返回单个结果.在SQL COUNT(*)和group by 相当于MongoDB的聚集. aggregat ...
- MongoDB聚合
--------------------MongoDB聚合-------------------- 1.aggregate(): 1.概念: 1.简介 ...
- MongoDB 聚合分组取第一条记录的案例及实现
关键字:MongoDB: aggregate:forEach 今天开发同学向我们提了一个紧急的需求,从集合mt_resources_access_log中,根据字段refererDomain分组,取分 ...
随机推荐
- python完成留言板功能
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="utf-8&quo ...
- 获取任意网站的图标,标题栏logo,网站logo
https://www.hao123.com/favicon.ico 网站换成你想要的 大多数都可以
- Codeforces Round #387 (Div. 2) 747F(数位DP)
题目大意 给出整数k和t,需要产生一个满足以下要求的第k个十六进制数 即十六进制数每一位上的数出现的次数不超过t 首先我们先这样考虑,如果给你了0~f每个数字可以使用的次数num[i],如何求长度为L ...
- [bzoj3065] 带插入区间第k小值 [重量平衡树套线段树]
题面 传送门 思路 发现强制在线了...... 本来可以树套树解决的问题,现在外层不能使用线段树了,拿什么替代呢? 我们需要一种支持单点插入.下套数据结构.数据结构上传合并复杂度最多单log,不能旋转 ...
- BZOJ3622 已经没有什么好害怕的了 【dp + 二项式反演】
题目链接 BZOJ3622 题解 既已开题 那就已经没有什么好害怕的了 由题目中奇怪的条件我们可以特判掉\(n - k\)为奇数时答案为\(0\) 否则我们要求的就是糖果大于药片恰好有\(\frac{ ...
- 牛客 2018NOIP 模你赛2 T2 分糖果 解题报告
分糖果 链接:https://www.nowcoder.com/acm/contest/173/B 来源:牛客网 题目描述 \(N\) 个小朋友围成一圈,你有无穷个糖果,想把其中一些分给他们. 从某个 ...
- hadoop学习之HDFS
1.什么是大数据?什么是云计算?什么是hadoop? 大数据现在很火,到底什么是大数据,多大的数据才算大,一般而言对于TB级以上的数据我们成为大数据,对于这些数据它的价值在哪?大数据的价值就是我们大量 ...
- Java面试题之hashmap中用什么hash算法解决碰撞的?
查了一下源码(jdk8),记录一下吧,能记住就记一下吧! static final int hash(Object key) { int h; return (key == null) ? 0 : ( ...
- HDU1710---树(知前序遍历与中序遍历 求后序遍历)
知前序遍历与中序遍历 求后序遍历 #include<iostream> #include<cstring> #include<queue> #include< ...
- Topcoder SRM 604 div1题解
CTSC考完跑了过来日常TC--- Easy(250pts): 题目大意:有个机器人,一开始的位置在(0,0),第k个回合可以向四个方向移动3^k的距离(不能不动),问是否可以到达(x,y),数据满足 ...