Java自动生成趋势比对数据的方法分享
作者:chz613 时间:2023-11-25 18:29:18
背景
数据之间两两趋势比较在数据分析应用中是非常常见的应用场景,如下所示:
模拟考批次 | 班级 | 学生 | 语文 | 数学 | 英语 |
---|---|---|---|---|---|
202302 | 三年一班 | 张小明 | 130 | 145 | 133 |
202302 | 三年一班 | 王二小 | 128 | 138 | 140 |
202302 | 三年一班 | 谢春花 | 136 | 142 | 139 |
202301 | 三年一班 | 张小明 | 132 | 140 | 128 |
202301 | 三年一班 | 王二小 | 125 | 146 | 142 |
202301 | 三年一班 | 谢春花 | 138 | 143 | 140 |
202212 | 三年一班 | 张小明 | 135 | 138 | 120 |
202212 | 三年一班 | 王二小 | 123 | 145 | 138 |
202212 | 三年一班 | 谢春花 | 136 | 140 | 142 |
现在有一个需求:各班级的每个学生在不同考试批次的各学科成绩的进退步情况,得出数据如下
模拟考批次 | 班级 | 学生 | 语文 | 数学 | 英语 |
---|---|---|---|---|---|
202302与202301对比 | 三年一班 | 张小明 | -2 | 5 | 5 |
202302与202301对比 | 三年一班 | 王二小 | 3 | -8 | -2 |
202302与202301对比 | 三年一班 | 谢春花 | -2 | -1 | -1 |
202301与202212对比 | 三年一班 | 张小明 | -3 | 2 | 8 |
202301与202212对比 | 三年一班 | 王二小 | 2 | 1 | 4 |
202301与202212对比 | 三年一班 | 谢春花 | 2 | 3 | -2 |
详细设计及实现
趋势比对定义类 TrendCompare
public class TrendCompare {
/**
* 主体的字段列表(如三年一班的张小明,那么主体字段列表为 班级 + 学生姓名)
*/
private String[] subjectFields;
/**
* 在某个字段的(介词) 如三年一班张晓明在不同考试批次的成绩对比结果
*/
private String atField;
/**
* 参与趋势比较的字段集合
*/
private String[] compareFields;
/**
* 赋值映射集合:给结果数据中指定的key设置指定的值
*/
private Map<String, String> assignValMap;
public String[] subjectFields() {
return this.subjectFields;
}
public TrendCompare subjectFields(String... fields) {
this.subjectFields = fields;
return this;
}
public String atField() {
return this.atField;
}
public TrendCompare atField(String field) {
this.atField = field;
return this;
}
public String[] compareFields() {
return this.compareFields;
}
public TrendCompare compareFields(String... fields) {
this.compareFields = fields;
return this;
}
/**
* 赋值操作
*
* @param field
* @param valueEL 值表达式
* @return
*/
public TrendCompare assignVal(String field, String valueEL) {
if (assignValMap == null) {
assignValMap = new HashMap<>();
}
assignValMap.put(field, valueEL);
return this;
}
public Map<String, String> assignValMap() {
return this.assignValMap;
}
}
该类定义了如下属性:
主体的字段列表
介词字段
比对的字段列表
如:各班级的每个学生在不同考试批次的各学科成绩的进退步情况
上面的需求映射到定义类的结果如下:
主体的字段列表(班级、学生)
介词字段(考试批次)
比对的字段列表(各学科:语文、数学、英语)
趋势比对执行类
该类提供了一个供外部调用的方法如下
public static <T> List<T> compare(List<T> dataList, TrendCompare trendCompare) {
Map<String, List<T>> groupMap = group(dataList, null, trendCompare.subjectFields());
List<T> resultList = new ArrayList<>();
for (List<T> groupDataList : groupMap.values()) {
List<T> diffValueList = new ArrayList<>();
int size = groupDataList.size();
if (size > 1) {
for (int i = 0; i < size - 1; i++) {
//数据之间两两比较 diffValue = minuend - subtrahend
T minuend = groupDataList.get(i);
T subtrahend = groupDataList.get(i + 1);
T diffValue = minus(trendCompare.compareFields(), minuend, subtrahend);
//设置主体信息
if (trendCompare.subjectFields() != null) {
for (String subjectField : trendCompare.subjectFields()) {
setFieldValue(diffValue, subjectField, getFieldValue(minuend, subjectField));
}
}
//设置介词字段信息
String atField = trendCompare.atField();
if (StringUtils.isNotEmpty(atField)) {
setFieldValue(diffValue, atField, getFieldValue(minuend, atField) + "与" + getFieldValue(subtrahend, atField) + "对比增减");
}
diffValueList.add(diffValue);
}
}
if (diffValueList.size() > 0) {
T firstData = groupDataList.get(0);
Map<String, Object> valMap = new HashMap<>();
//指定的赋值集合进行赋值
if (trendCompare.assignValMap() != null) {
for (Map.Entry<String, String> stringStringEntry : trendCompare.assignValMap().entrySet()) {
String field = stringStringEntry.getKey();
if (!StringUtils.equalsAny(field, trendCompare.compareFields())) {
String valueEL = stringStringEntry.getValue();
valMap.put(field, executeSpEL(valueEL, firstData));
}
}
}
for (Map.Entry<String, Object> entry : valMap.entrySet()) {
for (T diffValue : diffValueList) {
setFieldValue(diffValue, entry.getKey(), entry.getValue());
}
}
}
resultList.addAll(diffValueList);
}
return resultList;
}
可以看到,该方法要求传入
数据集合
趋势比对定义
两个参数,并最终返回趋势比对后的结果集合。
该方法的内部逻辑可分为如下2个步骤:
按主体分组
分组后组内数据两两比对,并最终返回比对结果。
使用案例
假设有如下这样一组数据
定义一个学生类:
public class Student {
private String batch;
private String banji;
private String studentNo;
private String name;
private String sex;
private Double yuwen;
private Double math;
private Double english;
private Double physics;
//extra
private String maxScoreName1;
public Student(String batch, String banji, String studentNo, String name, String sex, Double yuwen, Double math, Double english, Double physics) {
this.batch = batch;
this.banji = banji;
this.studentNo = studentNo;
this.name = name;
this.sex = sex;
this.yuwen = yuwen;
this.math = math;
this.english = english;
this.physics = physics;
}
}
我们写一个方法,返回如上数据:
public List<Student> getDataList() {
List<Student> dataList = new ArrayList<>();
dataList.add(new Student("202302", "三年一班", "20001001", "张小明", "男", 130.0, 145.0, 133.0, 92.0));
dataList.add(new Student("202302", "三年一班", "20001002", "王二小", "男", 128.0, 138.0, 140.0, 98.0));
dataList.add(new Student("202302", "三年一班", "20001003", "谢春花", "女", 136.0, 142.0, 139.0, 95.0));
dataList.add(new Student("202302", "三年二班", "20002001", "冯世杰", "男", 129.0, 144.0, 138.0, 96.0));
dataList.add(new Student("202302", "三年二班", "20002002", "马功成", "男", 130.0, 132.0, 133.0, 98.0));
dataList.add(new Student("202302", "三年二班", "20002003", "魏翩翩", "女", 136.0, 142.0, 137.0, 92.0));
dataList.add(new Student("202301", "三年一班", "20001001", "张小明", "男", 132.0, 142.0, 134.0, 92.0));
dataList.add(new Student("202301", "三年一班", "20001002", "王二小", "男", 126.0, 136.0, 135.0, 94.0));
dataList.add(new Student("202301", "三年一班", "20001003", "谢春花", "女", 136.0, 145.0, 139.0, 95.0));
dataList.add(new Student("202301", "三年二班", "20002001", "冯世杰", "男", 124.0, 143.0, 148.0, 90.0));
dataList.add(new Student("202301", "三年二班", "20002002", "马功成", "男", 140.0, 133.0, 138.0, 90.0));
dataList.add(new Student("202301", "三年二班", "20002003", "魏翩翩", "女", 126.0, 136.0, 135.0, 92.0));
return dataList;
}
趋势比对定义并执行比对:
List<Student> dataList = getDataList();
TrendCompare trendCompare = new TrendCompare()
.subjectFields("banji", "name")
.atField("batch")
.compareFields("yuwen", "math", "english")
//.assignVal("batch", "'环比增减'")
;
List<Student> resultList = DataProcessUtil.compare(dataList, trendCompare);
for (Student result : resultList) {
System.out.println(JSON.toJSONString(result));
}
结果如下:
{"banji":"三年一班","batch":"202302与202301对比增减","english":-1.0,"math":3.0,"name":"张小明","yuwen":-2.0}
{"banji":"三年一班","batch":"202302与202301对比增减","english":5.0,"math":2.0,"name":"王二小","yuwen":2.0}
{"banji":"三年一班","batch":"202302与202301对比增减","english":0.0,"math":-3.0,"name":"谢春花","yuwen":0.0}
{"banji":"三年二班","batch":"202302与202301对比增减","english":-10.0,"math":1.0,"name":"冯世杰","yuwen":5.0}
{"banji":"三年二班","batch":"202302与202301对比增减","english":-5.0,"math":-1.0,"name":"马功成","yuwen":-10.0}
{"banji":"三年二班","batch":"202302与202301对比增减","english":2.0,"math":6.0,"name":"魏翩翩","yuwen":10.0}
来源:https://juejin.cn/post/7218559384452087868
标签:Java,自动,生成,数据
0
投稿
猜你喜欢
mybatis初始化SqlSessionFactory失败的几个原因分析
2021-06-17 11:13:44
WPF实现绘制扇形统计图的示例代码
2021-11-29 23:42:00
学习使用Android Chronometer计时器
2022-02-20 16:58:25
JAVA使用commos-fileupload实现文件上传与下载实例解析
2022-07-08 03:47:13
Android三种方式实现ProgressBar自定义圆形进度条
2021-09-15 11:19:43
Java如何生成4位、6位随机数短信验证码(高效实现)
2023-01-24 18:05:20
Java数字格式类(NumberFormat类和DecimalFormat类)用法详解
2022-11-14 14:54:02
使用java实现telnet-client工具分享
2023-10-18 10:56:59
springboot相关面试题汇总详解
2023-10-06 17:16:11
Winform利用分页控件实现导出PDF文档功能
2023-08-11 10:15:04
JavaEE7+Websockets+GlassFish4打造聊天室
2023-11-29 01:01:39
java 使用JDBC构建简单的数据访问层实例详解
2023-01-13 05:12:24
Android 复制文本内容到系统剪贴板的最简单实例(分享)
2023-05-27 20:02:15
Android 使用flow实现倒计时的方式
2023-04-13 10:36:29
Java中如何调用cmd压缩文件
2023-12-09 13:37:02
java实现文件上传下载功能
2021-11-26 17:19:05
Java基于servlet监听器实现在线人数监控功能的方法
2021-08-19 11:38:24
.net(c#)中的new关键字详细介绍
2021-08-29 21:19:27
SpringBoot项目鉴权的4种方式小结
2021-10-23 20:10:05
Android Fragment概述及用法
2022-08-23 15:54:22