基于 MongoDB 动态字段设计的探索
一、业务需求
假设某学校课程系统,不同专业课程不同 (可以动态增删),但是需要根据专业不同显示该专业学生的各科课程的成绩,如下:
| 专业 | 姓名 | 高等数学 | 数据结构 |
|---|---|---|---|
| 计算机 | 张三 | 90 | 85 |
| 计算机 | 李四 | 78 | 87 |
| 专业 | 姓名 | 高等数学 |
|---|---|---|
| 数学 | 王五 | 86 |
| 数学 | 赵六 | 95 |
二、设计思路
开始的思路是根据配置的课程动态生成文档字段,使用非映射方式直接操作 MongoCollection, 有以下问题:
- 存取数据日期序列化问题 (亦可能是本人没有找到正确的处理方式)
- 返回结果集不能转换成实体对象,不方便做二次处理
所以最终使用内嵌数组的方式
三、代码示例
3.1 实体类
public class Student {
private String name;
private String major;
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
private LocalDateTime createTime;
@JsonIgnore
private List<Course> courseList;
/* getter setter*/
}
public class Course {
private String name;
private Float score;
/* getter setter*/
}
3.2 添加测试专业及课程
public class MajorConfig {
private static MajorConfig computer;
private static MajorConfig math;
static {
computer = new MajorConfig();
computer.setName("计算机");
List<String> courseList = new ArrayList<>();
courseList.add("高等数学");
courseList.add("数据结构");
computer.setCourseList(courseList);
math = new MajorConfig();
math.setName("数学");
courseList = new ArrayList<>();
courseList.add("高等数学");
math.setCourseList(courseList);
}
private String name;
private List<String> courseList;
/* getter setter*/
}
3.3 添加测试数据到 MongoDB
初始化数据 (createTime 在 MongoDB 存储为 ISODate):
[
{
"name": "张三",
"major": "计算机",
"createTime": "2018-01-20 08:00:00",
"courseList": [
{
"name": "高等数学",
"score": 20.283026
},
{
"name": "数据结构",
"score": 30.612194
}
]
},
{
"name": "王五",
"major": "数学",
"createTime": "2018-01-20 08:00:00",
"courseList": [
{
"name": "高等数学",
"score": 91.78229
}
]
},
{
"name": "李四",
"major": "计算机",
"createTime": "2019-10-01 07:10:50",
"courseList": [
{
"name": "高等数学",
"score": 60.488556
},
{
"name": "数据结构",
"score": 80.66098
}
]
},
{
"name": "赵六",
"major": "数学",
"createTime": "2019-10-01 07:10:50",
"courseList": [
{
"name": "高等数学",
"score": 29.595625
}
]
}
]
3.4 根据专业获取学生课程分数列表
首先查询到对应的数据,然后根据配置的课程动态添加字段:
public Object list(String major){
MajorConfig majorConfig = getMajorConfig(major);
Query query = new Query(Criteria.where("major").is(major));
List<Student> studentList = mongoTemplate.find(query, Student.class);
List<Object> result = new ArrayList<>();
for(Student student:studentList){
Map<String,Object> properties = Maps.newHashMap();
for(String name : majorConfig.getCourseList()){
properties.put(name,null);
for(Course course : student.getCourseList()){
if(name.equals(course.getName())){
properties.put(name,course.getScore());
break;
}
}
}
result.add(ReflectUtil.getTarget(student,properties));
}
return result;
}
各专业学生各科分数数据:
[
{
"name": "王五",
"major": "数学",
"createTime": "2018-01-20 08:00:00",
"高等数学": 91.78229
},
{
"name": "赵六",
"major": "数学",
"createTime": "2019-10-01 07:10:50",
"高等数学": 29.595625
}
]
[
{
"name": "张三",
"major": "计算机",
"createTime": "2018-01-20 08:00:00",
"高等数学": 20.283026,
"数据结构": 30.612194
},
{
"name": "李四",
"major": "计算机",
"createTime": "2019-10-01 07:10:50",
"高等数学": 60.488556,
"数据结构": 80.66098
}
]
ReflectUtil 代码:
public class ReflectUtil {
public static Object getTarget(Object dest, Map<String, Object> addProperties) {
PropertyUtilsBean propertyUtilsBean = new PropertyUtilsBean();
PropertyDescriptor[] descriptors = propertyUtilsBean.getPropertyDescriptors(dest);
Map<String, Class> propertyMap = Maps.newHashMap();
for (PropertyDescriptor d : descriptors) {
if (!"class".equalsIgnoreCase(d.getName())) {
propertyMap.put(d.getName(), d.getPropertyType());
}
}
// add extra properties
addProperties.forEach((k, v) -> {
if(v != null){
propertyMap.put(k, v.getClass());
}else{
propertyMap.put(k, Object.class);
}
});
// new dynamic bean
DynamicBean dynamicBean = new DynamicBean(dest.getClass(), propertyMap);
// add old value
propertyMap.forEach((k, v) -> {
try {
// filter extra properties
if (!addProperties.containsKey(k)) {
dynamicBean.setValue(k, propertyUtilsBean.getNestedProperty(dest, k));
}
} catch (Exception e) {
e.printStackTrace();
}
});
// add extra value
addProperties.forEach((k, v) -> {
try {
dynamicBean.setValue(k, v);
} catch (Exception e) {
e.printStackTrace();
}
});
Object target = dynamicBean.getTarget();
return target;
}
}
DynamicBean 代码:
public class DynamicBean {
private Object target;
private BeanMap beanMap;
public DynamicBean(Class superclass, Map<String, Class> propertyMap) {
this.target = generateBean(superclass, propertyMap);
this.beanMap = BeanMap.create(this.target);
}
public void setValue(String property, Object value) {
beanMap.put(property, value);
}
public Object getValue(String property) {
return beanMap.get(property);
}
public Object getTarget() {
return this.target;
}
private Object generateBean(Class superclass, Map<String, Class> propertyMap) {
BeanGenerator generator =new BeanGenerator();
if(null != superclass) {
generator.setSuperclass(superclass);
}
BeanGenerator.addProperties(generator, propertyMap);
return generator.create();
}
}
3.5 添加课程并修改数据库
public Object add(String major, String name, Boolean updateDatabase){
MajorConfig majorConfig = getMajorConfig(major);
List<String> courseList = majorConfig.getCourseList();
if(!courseList.contains(name))
{
courseList.add(name);
majorConfig.setCourseList(courseList);
}
if(updateDatabase){
Random random = new Random();
Course course = new Course();
course.setName(name);
course.setScore(random.nextFloat() * 100);
Update update = new Update();
update.addToSet("courseList", course);
Query query = Query.query(Criteria.where("major").is(majorConfig.getName()));
mongoTemplate.updateMulti(query, update, Student.class);
}
return majorConfig;
}
为数学专业添加计算机基础:
[
{
"name": "王五",
"major": "数学",
"createTime": "2018-01-20 08:00:00",
"计算机基础": 9.042096,
"高等数学": 91.78229
},
{
"name": "赵六",
"major": "数学",
"createTime": "2019-10-01 07:10:50",
"计算机基础": 9.042096,
"高等数学": 29.595625
}
]
3.6 删除课程并修改数据库
public Object del(String major, String name, Boolean updateDatabase){
MajorConfig majorConfig = getMajorConfig(major);
List<String> courseList = majorConfig.getCourseList();
if(courseList.contains(name)){
courseList.remove(name);
majorConfig.setCourseList(courseList);
}
if(updateDatabase){
Update update = new Update();
update.pull("courseList", new BasicDBObject("name", name));
Query query = Query.query(Criteria.where("major").is(majorConfig.getName()));
mongoTemplate.updateMulti(query,update,Student.class);
}
return majorConfig;
}
把高等数学从计算机专业删除:
[
{
"name": "张三",
"major": "计算机",
"createTime": "2018-01-20 08:00:00",
"数据结构": 30.612194
},
{
"name": "李四",
"major": "计算机",
"createTime": "2019-10-01 07:10:50",
"数据结构": 80.66098
}
]
3.7 修改某学生的某课程分数
public Object update(String name, String courseName, Float score){
Query query = Query.query(Criteria.where("name").is(name).and("courseList.name").is(courseName));
Update update = new Update();
update.set("courseList.$.score", score);
mongoTemplate.updateFirst(query, update, Student.class);
return null;
}
完整代码:GitHub
基于 MongoDB 动态字段设计的探索的更多相关文章
- 基于 MongoDB 动态字段设计的探索 (二) 聚合操作
业务需求及设计见前文:基于 MongoDB 动态字段设计的探索 根据专业计算各科平均分 (总分.最高分.最低分) public Object avg(String major){ Aggregatio ...
- 如何在Spring Data MongoDB 中保存和查询动态字段
原文: https://stackoverflow.com/questions/46466562/how-to-save-and-query-dynamic-fields-in-spring-data ...
- MongoDB 进阶模式设计
原文链接:http://www.mongoing.com/mongodb-advanced-pattern-design 12月12日上午,TJ在开源中国的年终盛典会上分享了文档模型设计的进阶技巧,就 ...
- 适用于app.config与web.config的ConfigUtil读写工具类 基于MongoDb官方C#驱动封装MongoDbCsharpHelper类(CRUD类) 基于ASP.NET WEB API实现分布式数据访问中间层(提供对数据库的CRUD) C# 实现AOP 的几种常见方式
适用于app.config与web.config的ConfigUtil读写工具类 之前文章:<两种读写配置文件的方案(app.config与web.config通用)>,现在重新整理一 ...
- SOA实践之基于服务总线的设计
在上文中,主要介绍了SOA的概念,什么叫做“服务”,“服务”应该具备哪些特性.本篇中,我将介绍SOA的一种很常见的设计实践--基于服务总线的设计. 基于服务总线的设计 基于总线的设计,借鉴了计算机内部 ...
- 基于Mongodb的轻量级领域驱动框架(序)
混园子也有些年头了,从各个大牛那儿学了很多东西.技术这东西和中国的料理一样,其中技巧和经验,代代相传(这不是舌尖上的中国广告).转身回头一望,几年来自己也积累了一些东西,五花八门涉猎到各种方向,今日开 ...
- 基于Apriori算法的Nginx+Lua+ELK异常流量拦截方案 郑昀 基于杨海波的设计文档(转)
郑昀 基于杨海波的设计文档 创建于2015/8/13 最后更新于2015/8/25 关键词:异常流量.rate limiting.Nginx.Apriori.频繁项集.先验算法.Lua.ELK 本文档 ...
- 基于MongoDB.Driver的扩展
由于MongoDB.Driver中的Find方法也支持表达式写法,结合[通用查询设计思想]这篇文章中的查询思想,个人基于MongoDB扩展了一些常用的方法. 首先我们从常用的查询开始,由于MongoD ...
- solr 通过【配置、多值字段、动态字段】来解决文本表达式查询精确到句子的问题
一.Solr Multivalue field属性positionIncrementGap理解 分类:Lucene 2014-01-22 10:39阅读(3596)评论(0) 参考:http://ro ...
随机推荐
- C语言100题集合005-删除一维数组中所有相同的数,使之只剩一个
系列文章<C语言经典100例>持续创作中,欢迎大家的关注和支持. 喜欢的同学记得点赞.转发.收藏哦- 后续C语言经典100例将会以pdf和代码的形式发放到公众号 欢迎关注:计算广告生态 即 ...
- iOS面试高薪,进阶 你会这些呢嘛?(持续更新中)
这个栏目将持续更新--请iOS的小伙伴关注!做这个的初心是希望能巩固自己的基础知识,当然也希望能帮助更多的开发者! 基础>分析>总结 面试 iOS常见基础面试题(附参考答案) iOS底层原 ...
- python开发基础(一)-if条件判断,while循环,break,continue,
条件语句 (1)if 基本语句 if 条件 : 内部代码块 else: .... print() (2)if 嵌套 (3)if elif 语句 (4)if 1==1: pass # if不执行,pas ...
- 动态规划入门——动态规划与数据结构的结合,在树上做DP
本文由TechFlow原创,本博文仅作为知识点学习,不会用于任何商业用途! 今天我们来看一个有趣的问题,通过这个有趣的问题,我们来了解一下在树形结构当中做动态规划的方法. 这个问题题意很简单,给定一棵 ...
- maven 搭建私服
博客参考 https://www.cnblogs.com/luotaoyeah/p/3791966.html 整理纯为技术学习 1 . 私服简介 私服是架设在局域网的一种特殊的远程仓库,目的是代理远程 ...
- 13、form组件
Form介绍 我们之前在HTML页面中利用form表单向后端提交数据时,都会写一些获取用户输入的标签并且用form标签把它们包起来. 与此同时我们在好多场景下都需要对用户的输入做校验,比如校验用户是否 ...
- 7.1range函数和冒泡排序
range函数和冒泡排序 一.range函数 在Python开发应用中 range函数相当重要,也比较常用: 首先看range函数的原型: range(start, end, scan) 参数 ...
- VGA详解及色块碰撞示例
引言 VGA:video Graphics array,视频图形阵列,阴极射线显像管(CRT)显示器时代产物,很多老显卡.笔记本电脑.投影仪所用接口,已经比较过时. CRT是模拟设备,所以VGA也采用 ...
- JS简单介绍与简单的基本语法
1.JavaScirpt是一门编程语言,是为前端服务的一门语言. (1)基础语法 (2)数据类型 (3)函数 (4)面向对象 2.还涉及到BOM和DOM (1)BOM(操作浏览器的一些功能) (2)D ...
- txt文件覆盖恢复
1.txt文件恢复到之前保存的版本 2.电脑未重启 方式:如果你使用系统还原可以用"还原以前的版本"功能来轻松找回. 右击.txt文件-还原以前的版本-选择时间点-还原