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

【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;如果商家商品卖完了&#…

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…

3D场景建模工具

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

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

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

PyQt6 QDialogButtonBox组合按钮控件

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

【开源威胁情报挖掘1】引言 + 开源威胁情报挖掘框架 + 开源威胁情报采集与识别提取

基于开源信息平台的威胁情报挖掘综述 写在最前面摘要1 引言近年来的一些新型网络安全威胁类型挖掘网络威胁的情报信息威胁情报分类&#xff1a;内、外部威胁情报国内外开源威胁情报挖掘分析工作主要贡献研究范围和方法 2 开源威胁情报挖掘框架1. 开源威胁情报采集与识别2. 开源…

软件生命周期四个阶段SDLC

软件产品生命周期&#xff1a;指软件产品研发全部过程、活动和任务的结构框架。 产品的生命周期一般包括四个阶段&#xff1a;引入期、成长期、成熟期和衰退期&#xff0c;在不同的阶段中&#xff0c;市场对产品的反应不同&#xff0c;其销售特点不同&#xff0c;因而产品管理的…

mysql数据库的配置文件在哪里

可以搜索my.ini、或者my.cnf&#xff0c;看看在哪个地方。 例如&#xff0c;我在windows系统装的mysql 8.2版本&#xff0c;my.ini文件不在安装目录下&#xff0c;而在另外一个目录下。 我的安装目录是F:\Program Files\MySQL\MySQL Server 8.2&#xff0c;但my.ini文件在C:\Pr…

【Leetcode题单】(01 数组篇)刷题关键点总结01【数组的遍历】

【Leetcode题单】&#xff08;01 数组篇&#xff09;刷题关键点总结01【数组的遍历】&#xff08;4题&#xff09; Easy数组的遍历485. 最大连续 1 的个数 Easy495. 提莫攻击 Easy414. 第三大的数 Easy628. 三个数的最大乘积 Easy 大家好&#xff0c;这里是新开的LeetCode刷题系…

JavaScript学习-1

01 基础用法 //index.html <body><!-- 行内JS --><button type"button" onclick"alert(hello)">按钮</button><!--内部JS--><script type"text/javascript">alert("你好");</script><…

vscode问题:此扩展在此工作区中被禁用,因为其被定义为在远程扩展主机中运行

mac按shiftcommandp windows按ctrlshiftP&#xff1a; 将当前项目文件夹添加进去就ok了。

Elk-filebeat

前言 Elk&#xff1a;filebeat搜集日志工具和logstash相同 Filebeat是一个轻量级的日志收集工具&#xff0c;所使用的资源比logstash部署和启动时使用的资源更小 Filebeat可以运行在非Java环境&#xff0c;他可以代理logstash在非Java环境上收集日志 Filebeat无法实现数据的…

【带头学C++】----- 九、类和对象 ---- 9.1 类和对象的基本概念----(9.1.4---9.1.6)

目录 9.1.4 设计立方体类 ​编辑 9.1.5 成员函数在类的外部实现 9.1.6 类在其他源文件的实现步骤&#xff08;实现类在不同文件的实现&#xff0c;后续引出构造函数&#xff09; 注意:类定义在同文件testclass.h中&#xff0c;而testclass.cpp是用来实现&#xff08;声明&…

Unity 简单打包脚本

打包脚本 这个打包脚本适用于做demo&#xff0c;脚本放在Editor目录下 using System; using System.Collections; using System.Collections.Generic; using System.IO; using UnityEditor; using UnityEngine;public class BuildAB {[MenuItem("Tools/递归遍历文件夹下…

K210开发板之VSCode开发环境使用中添加或删除文件(编译失败时)需要注意事项

在最初开始接触&#xff0c;将VScode和编译环境搭载好后&#xff0c;就开始运行第一个程序了&#xff0c;为了后续方便开发测试&#xff0c;这里我自己对照官方提供的例子&#xff0c;自己调试&#xff0c;写了一个简单的文件系统 后续&#xff0c;所有关于开发的源文件都在...…