java 工具类: CompareUtils(比较对象字段值变化)

一、前言

   我们在工作中,可能会在日志中记录数据的变化情况或者在公共处理的数据增加一个日志页面,记录每次修改的变化。我们可以根据CompareUtils工具类比较数据前后发生了怎样的变化, 这样我们就可以知道数据做了哪些改变.

二、条件限制

在写这个通用方法时,我们应该考虑到以下几点:

(1)可以接收任何对象的比较,但比较的对象应该是同个对象;

(2)可以给字段进行一个备注,因为我们看到的最终内容,应该是一个中文名称;

(3)一个对象中,可以忽略某些字段进行比较,只要我需要的字段进行比较。

2.1创建比较接口(自定义注解)

package com.zyqok.utils.compare;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;/*** 字段标记注解** @author qsg* @since 2022/05/05*/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Compare {/*** 字段名称*/String value();
}

 2.2 CompareNode(比较类)

package com.sinosoft.springbootplus.partyMember.compare;
/*** @author qsg* @since 2022/05/05*/
public class CompareNode {/*** 字段*/private String fieldKey;/*** 字段值*/private Object fieldValue;/*** 字段名称*/private String fieldName;public String getFieldKey() {return fieldKey;}public void setFieldKey(String fieldKey) {this.fieldKey = fieldKey;}public Object getFieldValue() {return fieldValue;}public void setFieldValue(Object fieldValue) {this.fieldValue = fieldValue;}public String getFieldName() {return fieldName;}public void setFieldName(String fieldName) {this.fieldName = fieldName;}
}

2.3创建比较对象(比较同一个实体)

在字段上方加上我们的自定义注解 @Compare注解, 可以通过注解进行两个对象的字段值的比较。

package com.sinosoft.springbootplus.partyMember.exportExcel;import cn.hutool.crypto.SecureUtil;
import cn.hutool.crypto.symmetric.AES;
import com.alibaba.excel.annotation.ExcelIgnore;
import com.alibaba.excel.annotation.ExcelProperty;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.sinosoft.springbootplus.mybaitsextend.dict.annotation.Dict;
import com.sinosoft.springbootplus.partyMember.compare.Compare;
import com.sinosoft.springbootplus.util.KeyStoreUtils;
import com.sun.org.apache.xerces.internal.impl.dv.util.HexBin;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;/*** <pre>** </pre>** @Author qsg* @Date 2023/11/29 9:29* @Version 1.0**/
@Data
@Slf4j
public class PartyMemberDiffExcel {private static final long serialVersionUID = 1L;private Long id;@ExcelProperty(value = "序号",index = 0)private int sortNum;@ExcelProperty(value = "党员姓名",index = 1)@Compare("党员姓名")private String memberName;@ExcelProperty(value = "性别",index = 2)@Dict(name = "sex" , dataSource = Dict.Type.DB, target = "memberSex")@Compare("性别")private String memberSex;@ExcelProperty(value = "年龄",index = 3)@Compare("年龄")private String age;@ExcelProperty(value = "出生日期",index = 4)@Compare("出生日期")private String birthday;@ExcelProperty(value = "民族",index = 5)@Dict(name = "nation", dataSource = Dict.Type.DB, target = "nation")@Compare("民族")private String nation;@ExcelProperty(value = "籍贯",index = 6)@Compare("籍贯")private String nativePlace;@ExcelProperty(value = "身份证号",index = 7)@Compare("身份证")private String cardId;@ExcelProperty(value = "所属党组织",index = 8)@Dict(name = "part_org_name", dataSource = Dict.Type.SERVIE, target = "partyOrgId")@Compare("所属党组织")private String partyOrgId;@ExcelProperty(value = "学历",index = 9)@Dict(name = "education_type", dataSource = Dict.Type.DB, target = "degree")@Compare("学历")private String degree;@ExcelProperty(value = "参加工作时间",index = 10)@Compare("参加工作时间")private String workTime;@ExcelProperty(value = "加入党组织日期",index = 11)@Compare("加入党组织日期")private String joinTime;@ExcelProperty(value = "转为正式党员日期",index = 12)@Compare("转为正式党员日期")private String regularTime;@ExcelProperty(value = "党籍状态",index = 13)@Dict(name = "party_status", target = "partyStatus", dataSource = Dict.Type.DB)@Compare("党籍状态")private String partyStatus;@ExcelProperty(value = "行政职务",index = 14)@Dict(name = "amd_postion" , dataSource = Dict.Type.DB, target = "admPosition")@Compare("行政职务")private String admPosition;@ExcelProperty(value = "职称",index = 15)@Dict(name = "title_type" , dataSource = Dict.Type.DB, target = "positionalTitles")@Compare("职称")private String positionalTitles;@ExcelProperty(value = "工作在一线情况",index = 16)@Compare("工作在一线情况")private String workingConditions;@ExcelProperty(value = "手机号码",index = 17)@Compare("手机号码")private String mobileNo;
}

2.4CompareUtils(比较工具类)

package com.sinosoft.springbootplus.partyMember.compare;import org.apache.poi.ss.formula.functions.T;
import java.lang.reflect.Field;import java.util.*;/*** <pre>** </pre>** @Author qsg* @Date 2023/11/28 15:14* @Version 1.0**/
public class CompareUtils<T> {private static final String COMMA = ",";/*** 属性比较** @param source 源数据对象* @param target 目标数据对象* @return 对应属性值的比较变化*/public String compare(T source, T target) {return compare(source, target, null);}/*** 属性比较** @param source              源数据对象* @param target              目标数据对象* @param ignoreCompareFields 忽略比较的字段* @return 对应属性值的比较变化*/public String compare(T source, T target, List<String> ignoreCompareFields) {if (Objects.isNull(source) && Objects.isNull(target)) {return "";}Map<String, CompareNode> sourceMap = this.getFiledValueMap(source);Map<String, CompareNode> targetMap = this.getFiledValueMap(target);if (sourceMap.isEmpty() && targetMap.isEmpty()) {return "";}// 如果源数据为空,则只显示目标数据,不显示属性变化情况if (sourceMap.isEmpty()) {return doEmpty(targetMap, ignoreCompareFields);}// 如果源数据为空,则显示属性变化情况String s = doCompare(sourceMap, targetMap, ignoreCompareFields);if (!s.endsWith(COMMA)) {return s;}return s.substring(0, s.length() - 1);}private String doEmpty(Map<String, CompareNode> targetMap, List<String> ignoreCompareFields) {StringBuilder sb = new StringBuilder();Collection<CompareNode> values = targetMap.values();int size = values.size();int current = 0;for (CompareNode node : values) {current++;Object o = Optional.ofNullable(node.getFieldValue()).orElse("");if (Objects.nonNull(ignoreCompareFields) && ignoreCompareFields.contains(node.getFieldKey())) {continue;}if (o.toString().length() > 0) {sb.append("[" + node.getFieldName() + ":" + o + "]");if (current < size) {sb.append(COMMA);}}}return sb.toString();}private String doCompare(Map<String, CompareNode> sourceMap, Map<String, CompareNode> targetMap, List<String> ignoreCompareFields) {StringBuilder sb = new StringBuilder();Set<String> keys = sourceMap.keySet();int size = keys.size();int current = 0;for (String key : keys) {current++;CompareNode sn = sourceMap.get(key);CompareNode tn = targetMap.get(key);if (Objects.nonNull(ignoreCompareFields) && ignoreCompareFields.contains(sn.getFieldKey())) {continue;}String sv = Optional.ofNullable(sn.getFieldValue()).orElse("").toString();String tv = Optional.ofNullable(tn.getFieldValue()).orElse("").toString();// 只有两者属性值不一致时, 才显示变化情况if (!sv.equals(tv)) {sb.append(String.format("[%s:%s -> %s]", sn.getFieldName(), sv, tv));if (current < size) {sb.append(COMMA);}}}return sb.toString();}private Map<String, CompareNode> getFiledValueMap(T t) {if (Objects.isNull(t)) {return Collections.emptyMap();}Field[] fields = t.getClass().getDeclaredFields();if (Objects.isNull(fields) || fields.length == 0) {return Collections.emptyMap();}Map<String, CompareNode> map = new LinkedHashMap();for (Field field : fields) {Compare annotation = field.getAnnotation(Compare.class);if (Objects.isNull(annotation)) {continue;}field.setAccessible(true);try {String fieldKey = field.getName();CompareNode node = new CompareNode();node.setFieldKey(fieldKey);node.setFieldValue(field.get(t));node.setFieldName(annotation.value());map.put(field.getName(), node);} catch (IllegalArgumentException | IllegalAccessException e) {e.printStackTrace();}}return map;}
}

三、比较现在的对象和之前的对象的值的变化

代码实现

 public List<String> expDiffPartyMembers(PartyMemberDifferentParam partyMemberDifferentParam,HttpServletResponse response) throws IOException {//日志标题设置为常量final String TITLE = "修改党员基本信息服务";//查询出sys_log表中更新党员的数据信息List<SysLog> sysLogs = sysLogMapper.selectList(new QueryWrapper<SysLog>().eq("type", LogTypeEnum.UPDATE.getCode()).eq("title", TITLE).ge("request_time", partyMemberDifferentParam.getStarTime()).le("request_time", partyMemberDifferentParam.getEndTime()));List<PartyMemberDiffExcel> partyMemberDiffs = new ArrayList<>();// 创建SimpleDateFormat对象,指定日期时间格式SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");Gson gson = new Gson();for (SysLog sysLog : sysLogs) {//获取党员修改json数据SysLogParam sysLogParam = sysLogParamMapper.selectById(sysLog.getId());String param = sysLogParam.getParam();//将党员信息json转换为对象PartyMemberDiffDto partyMemberDiffDto = gson.fromJson(param, PartyMemberDiffDto.class);PartyMemberDiffExcel partyMember = new PartyMemberDiffExcel();partyMember.setId(partyMemberDiffDto.getId());partyMember.setMemberName(partyMemberDiffDto.getMemberName());partyMember.setMemberSex(partyMemberDiffDto.getMemberSex());partyMember.setAge(String.valueOf(partyMemberDiffDto.getAge()));partyMember.setBirthday(ObjectUtils.isNotEmpty(partyMemberDiffDto.getBirthday())? sdf.format(new Date(partyMemberDiffDto.getBirthday())) :"" );partyMember.setNation(partyMemberDiffDto.getNation());partyMember.setNativePlace(partyMemberDiffDto.getNativePlace());partyMember.setCardId(partyMemberDiffDto.getCardId());partyMember.setPartyOrgId(String.valueOf(partyMemberDiffDto.getPartyOrgId()));partyMember.setDegree(partyMemberDiffDto.getDegree());partyMember.setWorkTime(partyMemberDiffDto.getWorkTime()!=null? sdf.format(new Date(partyMemberDiffDto.getWorkTime())):"");partyMember.setJoinTime(partyMemberDiffDto.getJoinTime()!=null? sdf.format(new Date(partyMemberDiffDto.getJoinTime())):"");partyMember.setRegularTime(partyMemberDiffDto.getRegularTime()!=null?sdf.format(new Date(partyMemberDiffDto.getJoinTime())):"");partyMember.setPartyStatus(partyMemberDiffDto.getPartyStatus());partyMember.setAdmPosition(partyMemberDiffDto.getAdmPosition());partyMember.setPositionalTitles(partyMemberDiffDto.getPositionalTitles());partyMember.setWorkingConditions(partyMemberDiffDto.getWorkingConditions());partyMember.setMobileNo(partyMemberDiffDto.getMobileNo());//放到list中存贮partyMemberDiffs.add(partyMember);}//去重List<PartyMemberDiffExcel> collect = partyMemberDiffs.stream().distinct().collect(Collectors.toList());//存储修改之后的党员信息id(现在党员表中的信息)List<Long> memberIds = new ArrayList<>();//获取在时间段修改的党员信息for (PartyMemberDiffExcel partyMemberDiffExcel : collect) {memberIds.add(partyMemberDiffExcel.getId());}List<String> list  = new ArrayList<>();//获取改之后的党员信息id(现在党员表中的信息)List<PartyMember> partyMemberList = partyMemberDomain.getPartyMemberDiff(memberIds);//现在的党员List<PartyMemberDiffExcel> partyMemberDiffNows = new ArrayList<>();for (PartyMember partyMember:partyMemberList){PartyMemberDiffExcel partyMemberDiffExcel = new PartyMemberDiffExcel();partyMemberDiffExcel.setId(partyMember.getId());partyMemberDiffExcel.setMemberName(partyMember.getMemberName());partyMemberDiffExcel.setMemberSex(partyMember.getMemberSex());partyMemberDiffExcel.setAge(String.valueOf(partyMember.getAge()));partyMemberDiffExcel.setBirthday(ObjectUtils.isNotEmpty(partyMember.getBirthday())? sdf.format(partyMember.getBirthday()) :"" );partyMemberDiffExcel.setNation(partyMember.getNation());partyMemberDiffExcel.setNativePlace(partyMember.getNativePlace());partyMemberDiffExcel.setCardId(partyMember.getCardId());partyMemberDiffExcel.setPartyOrgId(String.valueOf(partyMember.getPartyOrgId()));partyMemberDiffExcel.setDegree(partyMember.getDegree());partyMemberDiffExcel.setWorkTime(partyMember.getWorkTime()!=null? sdf.format(partyMember.getWorkTime()):"");partyMemberDiffExcel.setJoinTime(partyMember.getJoinTime()!=null? sdf.format(partyMember.getJoinTime()):"");partyMemberDiffExcel.setRegularTime(partyMember.getRegularTime()!=null?sdf.format(partyMember.getJoinTime()):"");partyMemberDiffExcel.setPartyStatus(partyMember.getPartyStatus());partyMemberDiffExcel.setAdmPosition(partyMember.getAdmPosition());partyMemberDiffExcel.setPositionalTitles(partyMember.getPositionalTitles());partyMemberDiffExcel.setWorkingConditions(partyMember.getWorkingConditions());partyMemberDiffExcel.setMobileNo(partyMember.getMobileNo());partyMemberDiffNows.add(partyMemberDiffExcel);}
//循环比较两个对象值的变化for (int i = 0; i < partyMemberList.size(); i++) {String compare = new CompareUtils<PartyMemberDiffExcel>().compare(collect.get(i), partyMemberDiffNows.get(i));System.out.println(compare);list.add(compare);return list;}

结果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/193345.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ImportError: cannot import name ‘metadata‘ from ‘importlib‘

yolov8 编译问题 ImportError: cannot import name ‘metadata’ from ‘importlib’ 将 from importlib import metadata 更改为 import importlib_metadata as metadata

《明解C语言》第三版 (入门篇) 第十三章练习答案

练习13-1 //打开与关闭文件#include <stdio.h>int main() {FILE* fd;char str[128];printf("请输入你要打开的文件\n");scanf("%s", str);fd fopen("*str","r");//只读方式打开if (fd NULL){puts("无法打开abc这个文件&…

【WinForm.NET开发】创建 Windows 窗体应用

本文内容 创建项目创建应用程序运行应用程序 本文演示创建一个具有基于 Windows 的用户界面 (UI) 的简单 C# 应用程序。 1、创建项目 首先&#xff0c;创建 C# 应用程序项目。 项目类型随附了所需的全部模板文件&#xff0c;无需添加任何内容。 打开 Visual Studio。在“开…

vs 安装 qt qt扩展

1 安装qt 社区版 免费 Download Qt OSS: Get Qt Online Installer 2 vs安装 qt vs tools 3 vs添加 qt添加 bin/cmake.exe 路径 3.1 扩展 -> qt versions 3.2

【设计模式-4.1】行为型——观察者模式

说明&#xff1a;本文介绍设计模式中行为型设计模式中的&#xff0c;观察者模式&#xff1b; 商家与顾客 观察者模式属于行为型设计模式&#xff0c;关注对象的行为。以商家与顾客为例&#xff0c;商家有商品&#xff0c;顾客来购买商品&#xff0c;如果商家商品卖完了&#…

Vue3中teleport如何使用

Vue 3作为一种流行的JavaScript框架&#xff0c;一直以来都在努力提供更便捷、灵活的开发体验。本文将深入解析Teleport&#xff0c;包括其详细的使用方法、源码实现机制以及在实际项目中的应用场景。 一、Teleport是什么&#xff1f; Vue 3中的Teleport允许开发者将组件的内容…

Vue+ElementUI+C#前后端分离:监控长耗时任务的实践

想象一下&#xff0c;我们正在构建一个Web应用&#xff0c;需要实现一个数据报告的导出功能。这听起来很简单&#xff0c;不是吗&#xff1f;但是&#xff0c;随着深入开发&#xff0c;我们意识到导出过程比预期的要复杂和耗时得多。由于报告的数据量巨大&#xff0c;后端需要花…

PostgreSQL有意思的现象:支持不带列的表

1、前言 以前从没有试过建一张表&#xff0c;不带任何列。在PG中却支持这种语法。这是个什么鬼? 最近&#xff0c;把PG源码扒了下&#xff0c;简单浏览了下最近的一些merge。其中有一个fix&#xff1a; eeb0ebad79 ("Fix the initial sync tables with no columns.&qu…

〖大前端 - 基础入门三大核心之JS篇㊺〗- 定时器和延时器

说明&#xff1a;该文属于 大前端全栈架构白宝书专栏&#xff0c;目前阶段免费&#xff0c;如需要项目实战或者是体系化资源&#xff0c;文末名片加V&#xff01;作者&#xff1a;不渴望力量的哈士奇(哈哥)&#xff0c;十余年工作经验, 从事过全栈研发、产品经理等工作&#xf…

LeetCode 每日一题 2023/11/27-2023/12/3

记录了初步解题思路 以及本地实现代码&#xff1b;并不一定为最优 也希望大家能一起探讨 一起进步 目录 11/27 907. 子数组的最小值之和11/28 1670. 设计前中后队列11/29 2336. 无限集中的最小数字11/30 1657. 确定两个字符串是否接近12/1 2661. 找出叠涂元素12/2 1094. 拼车12…

本地缓存和分布式缓存

一、引言 在当今的大数据时代&#xff0c;数据缓存已成为提升应用性能和效率的重要策略。缓存能够降低数据访问延迟&#xff0c;提高系统响应速度&#xff0c;从而改善用户体验。根据存储位置和应用场景的不同&#xff0c;缓存技术分为本地缓存和分布式缓存两种。本文将详细介绍…

【Rust日报】2023-12-01 KCL v0.7 版本发布

RFC&#xff1a;在选择依赖项时使 Cargo 遵循最低支持的 Rust 版本 (MSRV) 概括内容是&#xff0c;为需要使用旧版本 Rust 的开发者提供了一条快乐之路&#xff0c;具体方法是&#xff1a; 在 Cargo 解析依赖关系时&#xff0c;优先选择与 MSRV&#xff08;最低支持的 Rust 版本…

如何使用PHPUnit编写一个PHP单元测试-简单的代码示例

在软件开发过程中&#xff0c;单元测试是一种重要的测试方法&#xff0c;可以确保代码的质量和可靠性。在PHP开发中&#xff0c;也可以通过编写单元测试来验证代码的正确性。下面将介绍一些编写PHP单元测试的基本步骤和常用工具。 首先&#xff0c;你需要一个PHP单元测试框架&…

3D场景建模工具

在线工具推荐&#xff1a; 三维数字孪生场景工具 - GLTF/GLB在线编辑器 - Three.js AI自动纹理化开发 - YOLO 虚幻合成数据生成器 - 3D模型在线转换 - 3D模型预览图生成服务 1. 什么是3D场景建模&#xff1f; 3D场景建模是一种通过计算机图形学技术&#xff0c;将现实世…

python查询、处理、批量存入数据

1、安装数据库连接器 首先需要安装一个数据库连接器&#xff0c;比如pymysql、pyodbc等&#xff0c;用于连接MySQL、SQL Server等不同的数据库。 安装命令如下 pip install PyMySQL2. 连接数据库 连接数据库需要先指定数据库的主机名、端口号、用户名和密码等信息。这些信息…

Vue组件分装之$attrs、$listener传递属性及事件

使用v-bind"$attrs"来将父组件的属性传递给自定义按钮 使用v-on"$listeners"将父组件的事件监听器传递给自定义按钮。 使用$slots获取父组件所有插槽以及作用域插槽对应的参数#[name]"scopeData" 这样&#xff0c;自定义按钮就能够直接响应父…

Java系列 之除字符串中的空格(trim())

我 | 在这里 &#x1f575;️ 读书 | 长沙 ⭐软件工程 ⭐ 本科 &#x1f3e0; 工作 | 广州 ⭐ Java 全栈开发&#xff08;软件工程师&#xff09; &#x1f383; 爱好 | 研究技术、旅游、阅读、运动、喜欢流行歌曲 ✈️已经旅游的地点 | 新疆-乌鲁木齐、新疆-吐鲁番、广东-广州…

【Matlab】如何快速入门一项新技能-以Matlab/Simulink入门为例

目录 1. 引言 2. 背景 3. 快速学习并完成开发 3.1 了解需求&#xff0c;知道要干什么 3.2 了解Matlab/Simulink基本功能 第一步&#xff0c;查看Matlab的中文网站中文网站https://www.ilovematlab.cn/resources/对Matlab/Simulink有了一个初步认识。 3.3 实现一个最简单…

uniapp 在app端 使用webview进行数据交互。

使用案例 1.app端(需要使用nvue) <template> <view class"webview-box"> <button style"z-index: 999;" click"handlePostMessage(app向url传值)">点击向url传值</button><web-view ref"webview" clas…

PyQt6 QDialogButtonBox组合按钮控件

锋哥原创的PyQt6视频教程&#xff1a; 2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~_哔哩哔哩_bilibili2024版 PyQt6 Python桌面开发 视频教程(无废话版) 玩命更新中~共计34条视频&#xff0c;包括&#xff1a;2024版 PyQt6 Python桌面开发 视频教程(无废话版…