Java对比对象修改前与修改后字段发生的变化

news2025/1/9 16:34:48
  1.   开发过程中,我们通常会对系统操作人对系统的操作进行记录,记录操作前后某个字段的变化,如下图  

 2.   提供一个工具类,可以比较一个对象里面,源对象,与修改后的对象,有哪些字段发生了改变, 第一步 


/**
 * @author qiankun.hu
 * @version 1.0.0
 * @createTime 2023年09月20日 17:00:00
 * @Description TODO
 */
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ColumnInfo {

    /**
     * 字段code
     */
    String columnCode() default "";

    /**
     * 字段名称
     */
    String columnName() default "";

    /**
     * 字段类型
     */
    String columnType() default "";
}

工具类代码

import com.example.demo.config.ColumnInfo;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.BeanUtils;

import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.sql.Timestamp;
import java.util.*;

/**
 * @author qiankun.hu
 * @version 1.0.0
 * @createTime 2023年09月20日 17:00:00
 * @Description TODO
 */
public class MyBeanUtils extends BeanUtils {

    /**
     * 比较两个实体属性值,返回一个map以有差异的属性名为key,value为一个list分别存obj1,obj2此属性名的值
     *
     * @param obj1      源值
     * @param obj2      最新值
     * @param ignoreArr 选择忽略比较的属性数组
     * @return 属性差异比较结果map
     * <p>
     * 后面考虑添加注解字段名和中文名称,已达到提示对应的中文信息
     */
    public static Map<String, List<Object>> compareFields(Object obj1, Object obj2, String[] ignoreArr) {
        try {
            Map<String, List<Object>> map = new HashMap<String, List<Object>>();
            List<String> ignoreList = null;
            if (ignoreArr != null && ignoreArr.length > 0) {
                // array转化为list
                ignoreList = Arrays.asList(ignoreArr);
            }
            if (obj1 == null) {
                Class clazz = obj2.getClass();
                // 获取object的属性描述
                PropertyDescriptor[] pds = Introspector.getBeanInfo(clazz,
                        Object.class).getPropertyDescriptors();
                // 这里就是所有的属性了
                for (PropertyDescriptor pd : pds) {
                    // 属性名
                    String name = pd.getName();
                    // 如果当前属性选择忽略比较,跳到下一次循环
                    if (ignoreList != null && ignoreList.contains(name)) {
                        continue;
                    }
                    // Field field = clazz.getDeclaredField(name);
                    Field field;
                    try {
                        field = clazz.getDeclaredField(name);
                    } catch (NoSuchFieldException e) {
                        try {
                            //此处用于解决继承导致的getDeclaredField不能直接获取父类属性的问题
                            field = clazz.getSuperclass().getDeclaredField(name);
                        } catch (NoSuchFieldException ee) {
                            //此处用于解决继承导致的getDeclaredField不能直接获取父类属性的问题
                            field = clazz.getSuperclass().getSuperclass().getDeclaredField(name);
                        }
                    }
                    ColumnInfo columnInfo = field.getAnnotation(ColumnInfo.class);
                    if (columnInfo != null) {
                        String columName = columnInfo.columnName();
                        String columnCode = columnInfo.columnCode();
                        String columnType = columnInfo.columnType();
                        name = name + "_" + columName + "_" + columnCode + "_" + columnType;
                    } else {
                        continue;
                    }
                    // get方法
                    Method readMethod = pd.getReadMethod();
                    Object o2 = readMethod.invoke(obj2);
                    if (o2 != null) {
                        List<Object> list = new ArrayList<Object>();
                        list.add(null);
                        list.add(o2);
                        map.put(name, list);
                    }
                }
            } else {
                // 只有两个对象都是同一类型的才有可比性
                if (obj1.getClass() == obj2.getClass()) {
                    Class clazz = obj1.getClass();
                    // 获取object的属性描述
                    PropertyDescriptor[] pds = Introspector.getBeanInfo(clazz,
                            Object.class).getPropertyDescriptors();
                    // 这里就是所有的属性了
                    for (PropertyDescriptor pd : pds) {
                        // 属性名
                        String name = pd.getName();
                        // 如果当前属性选择忽略比较,跳到下一次循环
                        if (ignoreList != null && ignoreList.contains(name)) {
                            continue;
                        }
                        //Field field = clazz.getDeclaredField(name);
                        Field field;
                        try {
                            field = clazz.getDeclaredField(name);
                        } catch (NoSuchFieldException e) {
                            try {
                                //此处用于解决继承导致的getDeclaredField不能直接获取父类属性的问题
                                field = clazz.getSuperclass().getDeclaredField(name);
                            } catch (NoSuchFieldException ee) {
                                //此处用于解决继承导致的getDeclaredField不能直接获取父类属性的问题
                                field = clazz.getSuperclass().getSuperclass().getDeclaredField(name);
                            }

                        }
                        ColumnInfo columnInfo = field.getAnnotation(ColumnInfo.class);
                        if (columnInfo != null) {
                            String columName = columnInfo.columnName();
                            String columnCode = columnInfo.columnCode();
                            String columnType = columnInfo.columnType();
                            name = name + "_" + columName + "_" + columnCode + "_" + columnType;
                        } else {
                            continue;
                        }
                        // get方法
                        Method readMethod = pd.getReadMethod();
                        // 在obj1上调用get方法等同于获得obj1的属性值
                        Object o1 = readMethod.invoke(obj1);
                        // 在obj2上调用get方法等同于获得obj2的属性值
                        Object o2 = readMethod.invoke(obj2);
                        if (o1 instanceof Timestamp) {
                            o1 = new Date(((Timestamp) o1).getTime());
                        }
                        if (o2 instanceof Timestamp) {
                            o2 = new Date(((Timestamp) o2).getTime());
                        }
                        if (o1 == null && o2 == null) {
                            continue;
                        } else if (o1 == null && o2 != null) {
                            List<Object> list = new ArrayList<Object>();
                            list.add(o1);
                            list.add(o2);
                            if (o2 instanceof String && StringUtils.isBlank(String.valueOf(o2))) {
                                o2 = null;
                            }
                            if (o2 != null) {
                                map.put(name, list);
                            }
                            continue;
                        }
                        // 比较这两个值是否相等,不等就可以放入map了
                        if (!o1.equals(o2)) {
                            List<Object> list = new ArrayList<Object>();
                            list.add(o1);
                            list.add(o2);
                            if(o2!=null){
                                o1 = o1 instanceof String && StringUtils.isBlank(String.valueOf(o1)) ? "" : o1;
                                Object info = (o2 instanceof String && StringUtils.isBlank(String.valueOf(o2)) ? "" : o2);
                                o2 = o2 == null ? "" : info;
                                if(!o1.equals(o2)){
                                    map.put(name, list);
                                }
                            }

                        }
                    }
                }
            }
            return map;
        } catch (Exception e) {
            return null;
        }
    }
}

3  进行测试

import com.alibaba.fastjson.JSON;
import com.example.demo.config.ColumnInfo;
import com.example.demo.util.MyBeanUtils;
import java.util.*;


/**
 * @author qiankun.hu
 * @version 1.0.0
 * @createTime 2022年10月14日 13:35:00
 * @Description TODO
 */
public class StudentDto {

    @ColumnInfo(columnCode = "name", columnName = "名称")
    private String name;
    @ColumnInfo(columnCode = "name", columnName = "名称")
    private String age;
    @ColumnInfo(columnCode = "studentNum", columnName = "学号")
    private Integer studentNum;
    @ColumnInfo(columnCode = "classNum", columnName = "班级号")
    private String classNum;

    public static void main(String[] args) {
        //老数据
        StudentDto s1 = new  StudentDto();
        s1.setName("张三");
        s1.setAge("18");
        s1.setStudentNum(9001);
        s1.setClassNum("高一二班");

        //修改后的数据
        StudentDto s2 = new  StudentDto();
        s2.setName("张三");
        s2.setAge("19");
        s2.setStudentNum(9002);
        s2.setClassNum("高一三班");

        //忽略对比的字段
        String[] arr = new String[1];
        arr[0] = "studentNum";

        //对比哪些字段发生改变,K 是发生改变的字段  value 是字段前后的变化
        Map<String, List<Object>> stringListMap = MyBeanUtils.compareFields(s1, s2, null);

        System.out.println( JSON.toJSONString(stringListMap) );
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAge() {
        return age;
    }

    public void setAge(String age) {
        this.age = age;
    }

    public Integer getStudentNum() {
        return studentNum;
    }

    public void setStudentNum(Integer studentNum) {
        this.studentNum = studentNum;
    }

    public String getClassNum() {
        return classNum;
    }

    public void setClassNum(String classNum) {
        this.classNum = classNum;
    }
}

输出结果如下:
{
    "age_名称_name_": ["18", "19"],
    "studentNum_学号_studentNum_": [9001, 9002],
    "classNum_班级号_classNum_": ["高一二班", "高一三班"]
}

我们可以看到,对象修改后与修改后,相关字段产生的变化,如果有不需要对比的字段,传入相同的参数既可

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

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

相关文章

地球系统模式(CESM)应用及进阶

目前通用地球系统模式&#xff08;Community Earth System Model&#xff0c;CESM&#xff09;在研究地球的过去、现在和未来的气候状况中具有越来越普遍的应用。CESM由美国NCAR于2010年07月推出以来&#xff0c;一直受到气候学界的密切关注。近年升级的CESM2.0在大气、陆地、海…

在pandas中使matplotlib动态画子图的两种方法【推荐gridspec】

先上对比图&#xff0c; 第一种方法&#xff0c;这里仅展示1个大区&#xff0c;多个的话需要加一层循环就可以了&#xff0c;主要是看子图的画法 当大区下面的国家为1个或2个时&#xff0c;会进行报错 # 获取非洲国家列表 african_countries df[df[大区] 南亚大区][进口国…

ARM DAY3

硬件模块与总线连接&#xff1a;各种硬件模块&#xff08;如 GPIO 控制器&#xff09;与 CPU&#xff08;或内核&#xff09;通过总线进行连接。这个总线负责数据和指令的传输。 特殊功能寄存器&#xff08;SFRs&#xff09;的角色&#xff1a;每个硬件模块内部都有一组特殊功…

PowerDesigner 连接 MYSQL

我使用的是powerDesigner16的版本&#xff0c;使用前先保证安装了 mysql odbc 驱动包 选择&#xff1a;文件 -> 反向工程 -> database… 一大波图片正在来袭。。。 点击确认 至此连接成功

解决方案 | 如何构建市政综合管廊安全运行监测系统?

如何构建市政综合管廊安全运行监测系统&#xff1f;WITBEE万宾城市生命线智能监测仪器&#xff0c;5年免维护设计&#xff0c;集成10多项结构与气体健康监测指标&#xff0c;毫秒级快速响应&#xff0c;时刻感知综合管廊运行态势

make: /bin/nvcc: Command not found 解决找不到nvcc

首先输入nvcc -V查看nvcc路径&#xff0c;发现报错Command nvcc not found&#xff0c;接下来我们就要解决这个问题。 1&#xff09;进入cuda的bin目录cd /usr/local/cuda/bin&#xff0c;ls查看是否有nvcc 说明存在nvcc但cuda路径没有添加系统变量 2&#xff09;在bin目录下输…

C++中变量是按值访问的, Python 中变量的值是按引用访问的示例说明

C中变量是按值访问的&#xff0c; Python 中变量的值是按引用访问的示例说明 C、C、Python如何获取变量的地址 C语言 在C语言中&#xff0c;可以使用取地址运算符&#xff08;&&#xff09;来获取变量的地址。 例如&#xff1a; #include <stdio.h>int main() {…

基于SSM+Vue的网上超市系统设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用Vue技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…

苏宁滑块验证

网址&#xff1a;https://passport.suning.com/ids/login总结一下&#xff0c;别被他的表面现象给骗了&#xff0c;这玩意儿&#xff0c;个人认为&#xff0c;腾讯的都没法跟他比&#xff01;&#xff01;&#xff01; 难点&#xff1a;动态混淆&#xff0c;vmp&#xff0c;图片…

PID学习

文章目录 1、 简介2、P 比例调节3、I 积分控制4、D 微分控制5、简单的模拟PID输出代码6、改进6.1 采样时间6.1.1 问题所在6.1.2 解决方案6.1.3 代码6.1.4 结果6.1.5 关于中断的旁注6.1.6 个人总结 6.2 微分项出现尖峰6.2.1 问题所在6.2.2 解决方案6.2.3 代码6.2.4 结果6.2.5 个…

执行上下文,js、React、HTML中的this

目录 执行上下文属性&#xff1a;变量对象、this&#xff0c;作用域链 变量对象是与执行上下文相关的数据作用域&#xff0c;存储&#xff1a;变量、函数声明 执行上下文生命周期 创建&#xff1a;生成变量对象、创建函数作用域&#xff0c;建立作用域链、确定this的指向 …

jdk 21发布的意义

jdk 21 最大的功能是虚拟线程&#xff0c;是一种绿色线程&#xff08;具体可以看周志明老师的书籍《深入理解java虚拟机》&#xff09;&#xff0c;目前 jvm 与操作系统的线程是一一对应的关系。 使用了虚拟线程可以减少资源消耗&#xff0c;减少操作系统上下文切换&#xff0…

AIGC绘本——海马搬家来喽

随着ChatGPT的快速发展&#xff0c;人工智能领域也发生了翻天覆地的变化。今天&#xff0c;我们迎合科技潮流&#xff0c;利用AIGC的强大能力&#xff0c;可以创作很多精彩的作品&#xff0c;比如这样一本名为《海马搬家》的绘本&#xff08;注&#xff1a;此绘本根据同名儿童故…

strtok()函数的使用方法

strtok() 函数用于将字符串分割成子字符串&#xff08;标记&#xff09;。它在 C 语言中非常常用&#xff0c;可以通过指定分隔符来拆分原始字符串&#xff0c;并依次返回每个子字符串。 以下是 strtok() 函数的使用方法&#xff1a; #include <stdio.h> #include <…

RK3568平台开发系列讲解(驱动篇)RK3568 I2C总线介绍

🚀返回专栏总目录 文章目录 一、I2C 简介1.1、起始位1.2、停止位1.3、数据传输1.4、应答信号1.5、I2C 写时序1.6、I2C 读时序1.7、I2C 多字节读写时序二、RK3568 I2C 总线介绍沉淀、分享、成长,让自己和他人都能有所收获!😄 📢本篇我们将讲解RK3568 I2C总线特性。 一、…

【中文输入时没有了提示选项】

打开电脑时发现&#xff0c;输入中文时&#xff0c;下方一直没有出现提示文字的选项&#xff0c;可能是电脑自动更新兼容的问题&#xff0c;上网查询了解决方案&#xff0c;按照下方步骤可以得到解决&#xff1a; step1&#xff1a;window键i //打开设置窗口 step2&#xf…

软件项目测试用例评审

软件项目测试用例评审是确保测试计划的一部分&#xff08;即测试用例&#xff09;满足项目质量和要求的关键步骤之一。以下是一个通用的软件项目测试用例评审流程&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎…

UbuntuToGo | Ubuntu 22.04.6 VMware UEFI启动 VHD虚拟磁盘

Win下新建固定大小VHD磁盘 磁盘管理器 随便选一个磁盘 点击操作 创建固定大小VHD(多等一会儿 固定大小比较慢) VMware 中新建虚拟机 自定义 在选择磁盘之前全部默认&#xff0c;选择磁盘选择 使用现有虚拟磁盘 现有磁盘文件选择刚才新建的VHD文件(一定要是固定大小的VHD)…

在DFMEA实施过程中,如何区分预防措施和探测措施?

在FMEA分析中&#xff0c;我们常常需要在分析原因之后采取相应的改善措施&#xff0c;一般现行的控制方法有“预防”和“探测”两大类&#xff0c;但是很多情况下我们无法掌握两者的区别&#xff0c;在这里我们明确一下。 FMEA手册中对预防和探测是这样定义的&#xff1a; 预…

计网第五章(运输层)(八)(TCP的连接释放)

目录 一、基本概述 二、具体实现 三、经典问题之为什么客户进程不直接进入关闭状态&#xff1f; 四、保活计时器 一、基本概述 上篇博客&#xff08; 计网第五章&#xff08;运输层&#xff09;&#xff08;七&#xff09;&#xff08;TCP的连接建立&#xff09;&#xff…