以前用MongoDB数据库都是简单的查询,直接用Query就可以,最近项目中用到了分组查询,完全不一样。第一次遇到,搞了好几天终于有点那意思了。

先上代码:

 import java.math.BigDecimal;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List; import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.aggregation.Aggregation;
import org.springframework.data.mongodb.core.aggregation.AggregationResults;
import org.springframework.data.mongodb.core.aggregation.Fields;
import org.springframework.data.mongodb.core.aggregation.GroupOperation;
import org.springframework.data.mongodb.core.aggregation.MatchOperation;
import org.springframework.data.mongodb.core.aggregation.ProjectionOperation;
import org.springframework.data.mongodb.core.mapreduce.GroupBy;
import org.springframework.data.mongodb.core.mapreduce.GroupByResults;
import org.springframework.data.mongodb.core.query.Criteria;
import org.springframework.stereotype.Service; import com.mongodb.BasicDBList;
import com.mongodb.BasicDBObject;
import com.mongodb.CommandResult; @Service
public class EquipmentRepository implements EquipmentRepository{ private static final Logger logger = LoggerFactory.getLogger(EquipmentRepository.class); @Autowired
MongoTemplate mongoTemplate; /**
*<p>从登陆信息表中根据IP统计设备使用时间</p>
* @param hostName 设备名称
* @param startTime 统计开始时间
* @param endTime 统计结束时间
* @return 统计信息
*/
@Override
public List<EquipStatistics> statisticTime(String hostName, Date startTime, Date endTime) { List<EquipStatistics> equipStatisticsList = new ArrayList<EquipStatistics>(); try { String initial = "{hostName:'' ,equipmentTypeName:'', userDurateion : 0,count:0,"
+ "startTime:"+startTime.getTime()+",endTime:"+endTime.getTime()+",nowTime:"+new Date().getTime()+"}"; String reduceFunction = "function(doc,result){"
+ "if(doc.extraData.hostName) { result.hostName = doc.extraData.hostName;}"
+ "if(doc.extraData.deviceType) {result.equipmentTypeName = doc.extraData.deviceType;}"
+ "var time = doc.logoffTime.valueOf() - doc.logonTime.valueOf();"
+ "result.userDurateion +=time;"
+" result.count+=1;"
+ "}"; //时间的计算分四种情况
List<EquipStatistics> equipStatisticsListTemp =null;
for (int i = 0; i < 4; i++) {
switch (i) {
case 0:
//登出时间在开始和结束之间,登录在开始和结束之间的(登出-登录)
Criteria criteria = Criteria.where("logonIp").exists(true);
if(hostName !=null && !"".equals(hostName.trim())){
criteria.and("extraData.hostName").regex(hostName);
}
criteria.and("logoffTime").lt(endTime).gt(startTime).and("logonTime").lt(endTime).gt(startTime);
equipStatisticsListTemp = searchDB(criteria, reduceFunction, initial); break;
case 1:
//1、 登出时间为空或 登出时间在结束之后, 登录时间在开始与结束之间的(结束-登录)
reduceFunction = "function(doc,result){"
+ "if(doc.extraData.hostName) { result.hostName = doc.extraData.hostName;}"
+ "if(doc.extraData.deviceType) {result.equipmentTypeName = doc.extraData.deviceType;}"
+ "var time = result.endTime - doc.logonTime.valueOf();"
+ "result.userDurateion +=time;"
+" result.count+=1;"
+ "}";
Criteria criteria1 = Criteria.where("logonIp").exists(true);
if(hostName !=null && !"".equals(hostName.trim())){
criteria1.and("extraData.hostName").regex(hostName);
} criteria1.andOperator(Criteria.where("logonTime").lt(endTime).gt(startTime)
.andOperator(Criteria.where("logoffTime").exists(false).orOperator(Criteria.where("logoffTime").gt(endTime))));
equipStatisticsListTemp = searchDB(criteria1, reduceFunction, initial);
break;
case 2:
//2、 登出时间为空, 登出时间在结束之后 ,登录时间在开始之前的 (结束-开始)
reduceFunction = "function(doc,result){"
+ "if(doc.extraData.hostName) { result.hostName = doc.extraData.hostName;}"
+ "if(doc.extraData.deviceType) {result.equipmentTypeName = doc.extraData.deviceType;}"
+ "var time = result.endTime - result.startTime;"
+ "result.userDurateion +=time;"
+" result.count+=1;"
+ "}";
Criteria criteria2 = Criteria.where("logonIp").exists(true);
if(hostName !=null && !"".equals(hostName.trim())){
criteria2.and("extraData.hostName").regex(hostName);
}
criteria2.andOperator(Criteria.where("logonTime").lt(startTime)
.andOperator(Criteria.where("logoffTime").exists(false).orOperator(Criteria.where("logoffTime").gt(endTime))));
equipStatisticsListTemp = searchDB(criteria2, reduceFunction, initial);
break;
case 3:
//4、 登出时间在开始和结束之间,登录时间在开始之前的(登出-开始)
reduceFunction = "function(doc,result){"
+ "if(doc.extraData.hostName) { result.hostName = doc.extraData.hostName;}"
+ "if(doc.extraData.deviceType) {result.equipmentTypeName = doc.extraData.deviceType;}"
+ "var time = doc.logoffTime.valueOf() - result.startTime;"
+ "result.userDurateion +=time;"
+" result.count+=1;"
+ "}";
Criteria criteria3 = Criteria.where("logonIp").exists(true);
if(hostName !=null && !"".equals(hostName.trim())){
criteria3.and("extraData.hostName").regex(hostName);
}
criteria3.and("logonTime").lt(startTime).and("logoffTime").lt(endTime).gt(startTime);
equipStatisticsListTemp = searchDB(criteria3, reduceFunction, initial);
break;
default:
break;
}
equipStatisticsList.addAll(equipStatisticsListTemp);
equipStatisticsListTemp = null;
} //去除重复数据 时长相加 赋值使用率
equipStatisticsList = addDuration(equipStatisticsList,daysBetween(startTime,endTime));
} catch (Throwable e) {
logger.error("统计设备使用信息失败:"+e.getMessage(), e);
throw new AssetRuntimeException(e);
} return equipStatisticsList;
} //获取相隔天数
private int daysBetween(Date startTime, Date endTime) {
return (int)((endTime.getTime()-startTime.getTime())/(1000 * 86400));
} //查询数据库
private List<EquipStatistics> searchDB(Criteria criteria, String reduceFunction,
String initial) {
List<EquipStatistics> equipStatisticsList = new ArrayList<EquipStatistics>();
EquipStatistics equipStatistics = null;
GroupBy groupBy = GroupBy.key("logonIp")
.initialDocument(initial)
.reduceFunction(reduceFunction); GroupByResults<Session> results = mongoTemplate.group(criteria,
"sessions", groupBy, Session.class);
BasicDBList list = (BasicDBList)results.getRawResults().get("retval");
for (int i = 0; i < list.size(); i ++) {
equipStatistics = new EquipStatistics();
BasicDBObject obj = (BasicDBObject)list.get(i);
equipStatistics.setIp(obj.getString("logonIp"));
equipStatistics.setHostName(obj.getString("hostName"));
equipStatistics.setEquipmentTypeName(obj.getString("equipmentTypeName"));
equipStatistics.setUserDurateion(obj.getLong("userDurateion"));
equipStatisticsList.add(equipStatistics);
}
return equipStatisticsList;
} //去重
private List<EquipStatistics> addDuration(List<EquipStatistics> equipStatisticsList,int days) { BigDecimal base = new BigDecimal(days*8*60*60*1000+""); if(equipStatisticsList!=null){
for (int i = 0; i < equipStatisticsList.size()-1; i++) {
long userDurateion_i = equipStatisticsList.get(i).getUserDurateion();
equipStatisticsList.get(i).setUserdDurationStr(formatTime(userDurateion_i));
//
BigDecimal userDur_i = new BigDecimal(userDurateion_i);
double rate = userDur_i.divide(base, 4, BigDecimal.ROUND_HALF_UP).doubleValue();
equipStatisticsList.get(i).setUserRate(rate);
equipStatisticsList.get(i).setUserdRateStr(rate*100 + "%");
for(int j = equipStatisticsList.size()-1; j>i;j--){
long userDurateion_j = equipStatisticsList.get(j).getUserDurateion();
BigDecimal userDur_j = new BigDecimal(userDurateion_j);
rate = userDur_j.divide(base, 4, BigDecimal.ROUND_HALF_UP).doubleValue();
if(equipStatisticsList.get(i).getIp().equals(equipStatisticsList.get(j).getIp())){
equipStatisticsList.get(i).setUserDurateion(userDur_i.add(userDur_j).longValue());
equipStatisticsList.get(i).setUserdDurationStr(formatTime(userDur_i.add(userDur_j).longValue()));
rate = userDur_i.add(userDur_j).divide(base, 4, BigDecimal.ROUND_HALF_UP).doubleValue();
equipStatisticsList.get(i).setUserRate(rate);
equipStatisticsList.get(i).setUserdRateStr(rate*100 + "%");
equipStatisticsList.remove(j);
}else{
equipStatisticsList.get(j).setUserdDurationStr(formatTime(userDurateion_j));
equipStatisticsList.get(j).setUserRate(rate);
equipStatisticsList.get(j).setUserdRateStr(rate*100 + "%");;
}
}
}
}
return equipStatisticsList;
} /*
* 毫秒转化时分秒毫秒
*/
public String formatTime(Long ms) {
Integer ss = 1000;
Integer mi = ss * 60;
Integer hh = mi * 60;
Integer dd = hh * 24; Long day = ms / dd;
Long hour = (ms - day * dd) / hh;
Long minute = (ms - day * dd - hour * hh) / mi;
Long second = (ms - day * dd - hour * hh - minute * mi) / ss;
Long milliSecond = ms - day * dd - hour * hh - minute * mi - second * ss; StringBuffer sb = new StringBuffer();
if(day > 0) {
sb.append(day+"天");
}
if(hour > 0) {
sb.append(hour+"小时");
}
if(minute > 0) {
sb.append(minute+"分");
}
if(second > 0) {
sb.append(second+"秒");
}
if(milliSecond > 0) {
sb.append(milliSecond+"毫秒");
}
return sb.toString();
} //测试代码 public List getSessionTime() {
try {
CommandResult result = mongoTemplate.executeCommand("{aggregate : 'sessions', pipeline : "
+ "[{ $match : { logoffTime : {$exists:false} } },"
// + " { $group : { _id :logonIp,logonTime:{$sum:{logonTime.valueOf()}},logoffTime:{$sum:{logffTime.va}} } },"
+ " { $project : { _id : 0,logonHost : 1,logonIp : 1,logonTime : 1,extraData : 1,logoffTime : 1}}]}");
System.out.println(result); GroupBy groupBy = GroupBy.key("logonIp")
.initialDocument("{logonHost:'', sessionTime : 0, extraData : {}}")
.reduceFunction("function(doc,result){"
+ "result.logonHost = doc.logonHost;"
+ "var time = doc.logoffTime.valueOf() - doc.logonTime.valueOf();"
+ "result.sessionTime +=time ;"
+ "result.extraData = doc.extraData}");
GroupByResults<Session> results = mongoTemplate.group(Criteria.where("logoffTime").exists(true),
"sessions", groupBy, Session.class);
BasicDBList list = (BasicDBList)results.getRawResults().get("retval");
for (int i = 0; i < list.size(); i ++) {
BasicDBObject obj = (BasicDBObject)list.get(i);
System.out.println(obj.get("count"));
}
System.out.println(results);
}catch (Exception e) {
System.out.println(e);
}finally {
try{
MatchOperation matchOperation; matchOperation = new MatchOperation(Criteria.where("logonTime")
.lte(new SimpleDateFormat("yyyy-MM-dd").parse("2016-09-14"))
.gte(new SimpleDateFormat("yyyy-MM-dd").parse("2016-09-12"))
.andOperator(Criteria.where("logoffTime")
.lte(new SimpleDateFormat("yyyy-MM-dd").parse("2016-09-14"))
.gte(new SimpleDateFormat("yyyy-MM-dd").parse("2016-09-12")))
); GroupOperation groupOperation = new GroupOperation(Fields.fields("logonIp")); ProjectionOperation projectionOperation = new ProjectionOperation(Fields.fields("_id")); Aggregation aggregation = Aggregation.newAggregation(matchOperation,groupOperation,projectionOperation); AggregationResults<Object> groupResults
= mongoTemplate.aggregate(aggregation, "sessions", Object.class); List<Object> groupList = groupResults.getMappedResults();
for (Object object : groupList) {
System.out.println(object.toString());
}
} catch (ParseException e) {
e.printStackTrace();
}
}
return null;
} }

EquipmentRepository.java

//查询数据库
private List<EquipStatistics> searchDB(Criteria criteria, String reduceFunction,
String initial) {
List<EquipStatistics> equipStatisticsList = new ArrayList<EquipStatistics>();
EquipStatistics equipStatistics = null;
GroupBy groupBy = GroupBy.key("logonIp")
.initialDocument(initial)
.reduceFunction(reduceFunction); GroupByResults<Session> results = mongoTemplate.group(criteria,
"sessions", groupBy, Session.class);
BasicDBList list = (BasicDBList)results.getRawResults().get("retval");
for (int i = 0; i < list.size(); i ++) {
equipStatistics = new EquipStatistics();
BasicDBObject obj = (BasicDBObject)list.get(i);
equipStatistics.setIp(obj.getString("logonIp"));
equipStatistics.setHostName(obj.getString("hostName"));
equipStatistics.setEquipmentTypeName(obj.getString("equipmentTypeName"));
equipStatistics.setUserDurateion(obj.getLong("userDurateion"));
equipStatisticsList.add(equipStatistics);
}
return equipStatisticsList;
}

分组查询主要使用org.springframework.data.mongodb.core.mapreduce.GroupBy这个spring中的类:

例:

GroupBy groupBy = GroupBy.key("logonIp")
.initialDocument(initial)
.reduceFunction(reduceFunction);
GroupByResults<T> results = mongoTemplate.group(criteria,
"sessions", groupBy, T.class);

GroupBy.key('key'): key是所进行分组字段的字段名;

initial : 初始化对象,可理解为最后查询返回的数据初始化;

reduceFunction: js函数,用于对返回的结果进行处理操作;

function(doc,result){}:

doc是根据查询条件(相当于where条件)获取的每一条数据,result是最后的查询结果,初始值就是initial对象;

查询操作:

mongoTemplate.group(criteria,"session", groupBy, T.class);

criteria:相当于SQL中的where条件;

session: 数据库中的表名;

groupBy: -以上;

T.class: 这里是数据库表对应的domain

BasicDBList list = (BasicDBList)results.getRawResults().get("retval")

获取结果转为BasicDBList,"retval"是固定值,必须是它;

BasicDBObject obj = (BasicDBObject)list.get(i);  obj.getString("key");

key为initial中的key值,通过以上代码获取key值对应的value;

这只是其中一种用法......

MongoDB数据库GroupBy查询使用Spring-data-mongondb的实现的更多相关文章

  1. MongoDB数据库中查询数据(下)

    MongoDB数据库中查询数据(下) 在find中,options参数值为一个对象,用来设置查询数据时使用的选项,下面我们来对该参数值对象中可以使用的属性进行介绍: 1. fields; 该属性值为一 ...

  2. 在MongoDB数据库中查询数据(上)

    在MongoDB数据库中查询数据(上) 在MongoDB数据库中,可以使用Collection对象的find方法从一个集合中查询多个数据文档,find方法使用方法如下所示: collection.fi ...

  3. python 操作mongodb数据库模糊查询

    # -*- coding: utf-8 -*-import pymongoimport refrom pymongo import MongoClient #创建连接#10.20.66.106clie ...

  4. mongodb关联查询 和spring data mongodb

    GITHUB:https://github.com/peterowang/Springdata-mongo 使用DBRefs DBRefs中有三个字段 - $ref - 此字段指定引用文档的集合 $i ...

  5. mongodb数据库js查询

    #健康风险-disease db.disease.find({versions:'2'}).forEach(function(item){ item.diseaseDetail && ...

  6. 使用Spring访问Mongodb的方法大全——Spring Data MongoDB查询指南

    1.概述 Spring Data MongoDB 是Spring框架访问mongodb的神器,借助它可以非常方便的读写mongo库.本文介绍使用Spring Data MongoDB来访问mongod ...

  7. Spring Data MongoDB 一:入门篇(环境搭建、简单的CRUD操作)

    一.简介 Spring Data  MongoDB 项目提供与MongoDB文档数据库的集成.Spring Data MongoDB POJO的关键功能区域为中心的模型与MongoDB的DBColle ...

  8. 使用Spring访问Mongodb的方法大全——Spring Data MongoDB

    1.概述 Spring Data MongoDB 是Spring框架访问mongodb的神器,借助它可以非常方便的读写mongo库.本文介绍使用Spring Data MongoDB来访问mongod ...

  9. SpringBoot入门 (五) 数据库访问之spring data jpa

    本文记录学习使用spring data jpa访问数据库 一 什么是Spring Data JPA JPA(Java Persistence API)是Sun官方提出的Java持久化规范.它为Java ...

随机推荐

  1. python基础(五)缩进和选择

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 缩进 Python最具特色的是用缩进来标明成块的代码.我下面以if选择结构来举例. ...

  2. iOS打包Framework真机和模拟器兼容合并版本 - 详细攻略步骤

    打包Framework,测试时: 1.用模拟器打包,测试时只能跑在模拟器 2.用真机打包,测试时只能跑在真机 那么怎么做到一个版本兼容以上两种场景呢? 解决如下: 1.打开终端 2.输入   lipo ...

  3. my_strcpy()

    char* my_strcpy(char* des, const char* src){ while(*des++=*src++); return des; }

  4. LINUX运维实战案例之文件已删除但空间不释放问题的分析与解决办法

    1.错误现象 运维的监控系统发来通知,报告一台服务器空间满了,登陆服务器查看,根分区确实没有空间了,如下图所示: 这里首先说明一下服务器的一些删除策略,由于Linux没有回收站功能,我们的线上服务器所 ...

  5. Linux 下安装配置 JDK

    JDK 下载地址 http://www.oracle.com/technetwork/java/javase/downloads/index.html 按照自己的情况选择不同的版本下载 cd /usr ...

  6. windows文件关联、打开方式列表之修改注册表攻略

    这里全是修改注册表的方式.网上找了半天,有的仅有添加文件关联的方法,却没有添加到打开方式列表里面的方法:有的有添加到文件列表的方法,却是使 用控制面板->文件夹选项的.好难得才找齐所有,从添加文 ...

  7. cdoj 1489 老司机采花

    地址:http://acm.uestc.edu.cn/#/problem/show/1489 题目: 老司机采花 Time Limit: 3000/1000MS (Java/Others)     M ...

  8. [转]Asp.net mvc 网站之速度优化 -- 页面缓存

    网站速度优化的一般方法 由于网站最重要的用户体验就是速度,特别是对于电子商务网站而言. 一般网站速度优化会涉及到几个方面: 1. 数据库优化 — 查询字段简历索引,使用数据库连接池和持久化,现在还有种 ...

  9. Eclipse c++代码提示,覆盖下面代码的问题。

    今天在使用Eclipse自动提示时,会覆盖下面行的代码!!! 这个错误几乎不能忍,goolge无果. 手动尝试去掉,全部代码提示,终于找到解法办法,但是原因未知. 如下图:需要去掉 "Par ...

  10. uwsgi+flask环境中安装matplotlib

    uwsgi+flask的python有自身的virtual environment,可以通过如下命令进入 . venv/bin/activate 虽然通过sudo apt-get install py ...