当前位置: 首页 > news >正文

利用Java反射机制实现对象相同字段的复制

一。如何实现不同类型对象之间的复制问题?

1、为什么会有这个问题?

近来在进行一个项目开发的时候,为了隐藏后端数据库表结构、同时也为了配合给前端一个更友好的API接口文档(swagger API文档),我采用POJO来对应数据表结构,使用VO来给传递前端要展示的数据,同时使用DTO来进行请求参数的封装。以上是一个具体的场景,可以发现这样子一个现象:POJO、VO、DTO对象是同一个数据的不同视图,所以会有很多相同的字段,由于不同的地方使用不同的对象,无可避免的会存在对象之间的值迁移问题,迁移的一个特征就是需要迁移的值字段相同。字段相同,于是才有了不同对象之间进行值迁移复制的问题。

2、现有的解决方法

一个一个的get出来后又set进去。这个方法无可避免会增加很多的编码复杂度,还是一些很没有营养的代码,看多了还会烦,所以作为一个有点小追求的程序员都没有办法忍受这种摧残。
使用别人已经存在的工具。在spring包里面有一个可以复制对象属性的工具方法,可以进行对象值的复制,下一段我们详细去分析它的这个工具方法。
自己动手丰衣足食。自己造工具来用,之所以自己造工具不是因为喜欢造工具,而是现有的工具没办法解决自己的需求,不得已而为之。

二、他山之石可以攻玉,详谈spring的对象复制工具

1、看看spring的对象复制工具到底咋样?

类名:org.springframework.beans.BeanUtils
这个类里面所有的属性复制的方法都调用了同一个方法,我们就直接分析这个原始的方法就行了。

/**
 * Copy the property values of the given source bean into the given target bean.
 * <p>Note: The source and target classes do not have to match or even be derived
 * from each other, as long as the properties match. Any bean properties that the
 * source bean exposes but the target bean does not will silently be ignored.
 * @param source the source bean:也就是说要从这个对象里面复制值出去
 * @param target the target bean:出去就是复制到这里面来
 * @param editable the class (or interface) to restrict property setting to:这个类对象是target的父类或其实现的接口,用于控制属性复制的范围
 * @param ignoreProperties array of property names to ignore:需要忽略的字段
 * @throws BeansException if the copying failed
 * @see BeanWrapper
 */
private static void copyProperties(Object source, Object target, Class<?> editable, String... ignoreProperties)
        throws BeansException {

    //这里在校验要复制的对象是不可以为null的,这两个方法可是会报错的!!
    Assert.notNull(source, "Source must not be null");
    Assert.notNull(target, "Target must not be null");
    //这里和下面的代码就有意思了
    Class<?> actualEditable = target.getClass();//获取目标对象的动态类型
    //下面判断的意图在于控制属性复制的范围
    if (editable != null) {
        //必须是target对象的父类或者其实现的接口类型,相当于instanceof运算符
        if (!editable.isInstance(target)) {
            throw new IllegalArgumentException("Target class [" + target.getClass().getName() +
                    "] not assignable to Editable class [" + editable.getName() + "]");
        }
        actualEditable = editable;
    }
    //不得不说,下面这段代码乖巧的像绵羊,待我们来分析分析它是如何如何乖巧的
    PropertyDescriptor[] targetPds = getPropertyDescriptors(actualEditable);//获取属性描述,描述是什么?描述就是对属性的方法信息的封装,好乖。
    List<String> ignoreList = (ignoreProperties != null ? Arrays.asList(ignoreProperties) : null);

    //重头戏开始了!开始进行复制了
    for (PropertyDescriptor targetPd : targetPds) {
        //先判断有没有写方法,没有写方法我也就没有必要读属性出来了,这个懒偷的真好!
        Method writeMethod = targetPd.getWriteMethod();
        //首先,没有写方法的字段我不写,乖巧撒?就是说你不让我改我就不改,让我忽略我就忽略!
        if (writeMethod != null && (ignoreList == null || !ignoreList.contains(targetPd.getName()))) {
            PropertyDescriptor sourcePd = getPropertyDescriptor(source.getClass(), targetPd.getName());
            //如果没办法从原对象里面读出属性也没有必要继续了
            if (sourcePd != null) {
                Method readMethod = sourcePd.getReadMethod();
                //这里就更乖巧了!写方法不让我写我也不写!!!
                if (readMethod != null &&
                        ClassUtils.isAssignable(writeMethod.getParameterTypes()[0], readMethod.getReturnType())) {
                    try {
                        //这里就算了,来都来了,就乖乖地进行值复制吧,别搞东搞西的了
                        if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers())) {
                            readMethod.setAccessible(true);
                        }
                        Object value = readMethod.invoke(source);
                        if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers())) {
                            writeMethod.setAccessible(true);
                        }
                        writeMethod.invoke(target, value);
                    }
                    catch (Throwable ex) {
                        throw new FatalBeanException(
                                "Could not copy property '" + targetPd.getName() + "' from source to target", ex);
                    }
                }
            }
        }
    }
}

2、对复制工具的一些看法和总结

总结上一段代码的分析,我们发现spring自带的工具有以下特点:
它名副其实的是在复制属性,而不是字段!!
它可以通过一个目标对象的父类或者其实现的接口来控制需要复制属性的范围
很贴心的可以忽略原对象的某些字段,可以通过2的方法忽略某些目标对象的字段
但是,这远远不够!!!我需要如下的功能:
复制对象的字段,而不是属性,也就是说我需要一个更暴力的复制工具。
我需要忽略原对象的某些字段,同时也能够忽略目标对象的某些字段。
我的项目还需要忽略原对象为null的字段和目标对象不为null的字段
带着这三个需求,开始我的工具制造。


作者:树树在变干
来源:CSDN
https://3pomelos.1688.com/

转载于:https://blog.51cto.com/14127231/2334091

相关文章:

  • Markdown基本语法
  • 浅探前端图片优化
  • kubernetes 监控方案之:heapster+influxdb+grafana(十八)
  • Servlet-生命周期
  • CSS 文字太多用省略号表示
  • 互联网轻量级框架SSM-查缺补漏第四天
  • 无需Java代码通过JHipster生成有安全验证的微服务应用
  • BZOJ5341[Ctsc2018]暴力写挂——边分治+虚树+树形DP
  • JS变量作用域
  • android环境搭建
  • Windows10 VS2017 C++多线程传参和等待线程结束
  • ZooKeeper的安装与部署
  • Vue.js常用指令:v-on
  • 如何设计一个比特币钱包服务
  • JS 编译器都做了啥?
  • [PHP内核探索]PHP中的哈希表
  • [译]前端离线指南(上)
  • 2017 年终总结 —— 在路上
  • 2017 前端面试准备 - 收藏集 - 掘金
  • Angular 响应式表单之下拉框
  • go append函数以及写入
  • JavaScript中的对象个人分享
  • Java知识点总结(JDBC-连接步骤及CRUD)
  • JS进阶 - JS 、JS-Web-API与DOM、BOM
  • JS学习笔记——闭包
  • Leetcode 27 Remove Element
  • MobX
  • Nginx 通过 Lua + Redis 实现动态封禁 IP
  • PAT A1017 优先队列
  • Puppeteer:浏览器控制器
  • Webpack 4x 之路 ( 四 )
  • 从零开始的无人驾驶 1
  • 分享一份非常强势的Android面试题
  • 工作中总结前端开发流程--vue项目
  • 互联网大裁员:Java程序员失工作,焉知不能进ali?
  • 悄悄地说一个bug
  • 如何在GitHub上创建个人博客
  • 使用阿里云发布分布式网站,开发时候应该注意什么?
  • No resource identifier found for attribute,RxJava之zip操作符
  • 阿里云移动端播放器高级功能介绍
  • ###51单片机学习(1)-----单片机烧录软件的使用,以及如何建立一个工程项目
  • #if #elif #endif
  • #我与Java虚拟机的故事#连载09:面试大厂逃不过的JVM
  • $.ajax()方法详解
  • $.extend({},旧的,新的);合并对象,后面的覆盖前面的
  • (1)bark-ml
  • (独孤九剑)--文件系统
  • (附源码)ssm教材管理系统 毕业设计 011229
  • (机器学习的矩阵)(向量、矩阵与多元线性回归)
  • (剑指Offer)面试题34:丑数
  • (力扣)1314.矩阵区域和
  • (四)模仿学习-完成后台管理页面查询
  • (译)计算距离、方位和更多经纬度之间的点
  • (原創) 如何刪除Windows Live Writer留在本機的文章? (Web) (Windows Live Writer)
  • .equals()到底是什么意思?