点击蓝色“程序猿DD”关注我

回复“资源”获取独家整理的学习资料!

作者 | 绝色天龙

来源 | https://www.jianshu.com/p/357b55852efc

背景

最近项目中在和第三方进行联调一个接口,我们这边发送http请求给对方,然后接收对方的回应,代码都是老代码。根据注释,对方的SDK中写好的Request类有一个无法序列化的bug,所以这边重新写了一个Request类,基本属性都是相同的,但是重点是有一个属性是静态内部类,还有两个是list属性,类似于下面这样:

private List<Order> orders;
private AddRequest.Ticket ticket;
private List<Payment> payments;

AddRequest就是我们自己重写的请求类,他们SDK中的请求类是MixAddRequest,我们组装好请求参数后利用Spring的BeanUtils的copyProperties方法将AddRequest中的属性拷贝到MixAddRequest,然后发送请求。到此为止,照理说一切完美

结果请求失败,纳尼?对方说缺少一个必要的字段,参数校验不通过,一查字段名称,是Ticket这个类里面的某个字段,赶紧看代码,心里充满对老代码的自信,想着一定是哪里搞错了,或者是他们那边偷偷动了代码,把字段从可选改为了必选,嘿嘿

果然在代码里找到了设置的地方,这下应该是他们的问题确信无疑了,再开一把调试,准备宣判他们的死刑。结果发现发给他们的请求就是没有这个字段。。。中间只有一个Spring的copy属性的方法,当时觉得很诡异

由于中间只有这么一行代码,玄机肯定在这里面,初步怀疑是两个静态内部类不同导致,所以自己写Demo,准备搞一把这个BeanUtils的copyProperties方法,写了两个类和一个Main,@Data和@ToString是lombok插件的注解,这里用来自动生成getter和setter方法以及toString方法

@ToString
@Data
public classCopyTest1{public String outerName;public CopyTest1.InnerClass innerClass;public List<CopyTest1.InnerClass> clazz;@ToString@Datapublic static classInnerClass{public String InnerName;}
}
@ToString
@Data
public classCopyTest2{public String outerName;public CopyTest2.InnerClass innerClass;public List<CopyTest2.InnerClass> clazz;@ToString@Datapublic static classInnerClass{public String InnerName;}
}CopyTest1 test1 = new CopyTest1();test1.outerName = "hahaha";CopyTest1.InnerClass innerClass = new CopyTest1.InnerClass();innerClass.InnerName = "hohoho";test1.innerClass = innerClass;System.out.println(test1.toString());CopyTest2 test2 = new CopyTest2();BeanUtils.copyProperties(test1, test2);System.out.println(test2.toString());

这里遇到了第一个坑,一开始图省事,属性写为public,想着省掉了getter和setter方法,没加@Data注解,结果运行完test2所有属性都为null,一个都没copy过去,加上@Data继续跑,果然,基本属性(String)复制过去了,但是内部类在test2中还是null。那就验证了真的是内部类的问题,有点不敢相信自己的眼睛,毕竟线上跑了这么久的代码。。。

知道了问题,总要想着怎么解决吧,所以需要单独设置一下内部类,单独copy,如果内部类的bean属性较多或者递归的bean属性很多,那可以自己封装一个方法,用于递归拷贝,我这里只有一层,所以直接额外copy一次

        CopyTest1 test1 = new CopyTest1();test1.outerName = "hahaha";CopyTest1.InnerClass innerClass = new CopyTest1.InnerClass();innerClass.InnerName = "hohoho";test1.innerClass = innerClass;System.out.println(test1.toString());CopyTest2 test2 = new CopyTest2();test2.innerClass = new CopyTest2.InnerClass();BeanUtils.copyProperties(test1, test2);BeanUtils.copyProperties(test1.innerClass, test2.innerClass);System.out.println(test2.toString());

记得内部类的属性也是要有setter方法的,不然也会导致copy失败,大家还记得我开头说到还有两个List属性的吧,为什么要提到这个呢?你猜

其实list里面的两个类也都是重写的内部类,他们也是不同的,当时他们却顺利copy过去了,为什么呢?因为java的泛型只在编译期起作用,在运行期,list属性就是一个存放Object的集合,在copy后,MixAddRequest的orders属性其实是一个Order类的集合,但却不是自己内部类的集合,是AddRequest的内部类Order的集合,但因为对方是解析json的,所以没有发生错误。。。

总结

1. Spring的BeanUtils的CopyProperties方法需要对应的属性有getter和setter方法;

2. 如果存在属性完全相同的内部类,但是不是同一个内部类,即分别属于各自的内部类,则spring会认为属性不同,不会copy;

3. 泛型只在编译期起作用,不能依靠泛型来做运行期的限制;

4. 最后,spring和apache的copy属性的方法源和目的参数的位置正好相反,所以导包和调用的时候都要注意一下。

最后的最后

附上spring的源码,getWriteMethod是jdk的方法,会去取set开头的方法,所以没有setter方法是不行滴。

privatestaticvoid copyProperties(Object source, Object target, @Nullable Class<?> editable, @Nullable String... ignoreProperties) throws BeansException {Assert.notNull(source, "Source must not be null");Assert.notNull(target, "Target must not be null");Class<?> actualEditable = target.getClass();if (editable != null) {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;PropertyDescriptor[] var7 = targetPds;int var8 = targetPds.length;for(int var9 = 0; var9 < var8; ++var9) {PropertyDescriptor targetPd = var7[var9];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 var15) {throw new FatalBeanException("Could not copy property '" + targetPd.getName() + "' from source to target", var15);}}}}}}

本文通过OpenWrite的免费Markdown转换工具发布

-END-

留言交流不过瘾

关注我,回复“加群”加入各种主题讨论群

四条使用Spring BeanUtils的总结,避免各种诡异的属性拷贝问题!相关推荐

  1. 四条使用Spring BeanUtils的总结

    点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 作者 | 绝色天龙 来源 | jianshu.com/p ...

  2. 假如有Thread1、Thread2、ThreaD3、Thread4四条线程分别统计C、D、E、F四个盘的大小,所有线程都统计完毕交给Thread5线程去做汇总,应当如何实现?...

    有两种方法: 第一种方法: 一般情况,我们实现多线程都是Thread或者Runnable(后者比较多),但是,这两种都是没返回值的,所以我们需要使用callable(有返回值的多线程)和future( ...

  3. Nature:给博士研究生的四条箴言Four golden lessons,颜宁推荐

    导语 史蒂文·温伯格(Steven Weinberg,1933年5月3日-),生于纽约,美国物理学家,1979年获诺贝尔物理学奖. <Four golden lessons>是美国物理学家 ...

  4. 颜宁强烈推荐:给研究生的四条金玉良言

    点击上方"AI遇见机器学习",选择"星标"公众号 原创干货,第一时间送达 转自:岩之有理 史蒂文·温伯格(Steven Weinberg,1933年5月3日-) ...

  5. 画蛇添足:四条管脚的电位器

    01四端电位器 电位器(potentiometer)也被称作滑动变阻器(rheostat)是一个机械电子器件,通常具有三个端口,即左右两个固定端以及中间的滑动端. 两个固定端连接在 一个由电阻丝.碳膜 ...

  6. Nature:给博士研究生的四条箴言Four golden lessons,颜宁:写的真好!

    史蒂文·温伯格(Steven Weinberg,1933年5月3日-),生于纽约,美国物理学家,1979年获诺贝尔物理学奖. <Four golden lessons>是美国物理学家.诺贝 ...

  7. python预测新航线的票价_浙江长龙航空有限公司四条新航线(线路+票价)

    新航线开通时间:4月5日 新开通航线线路:杭州-郑州.杭州-洛阳-成都.杭州-赤峰-哈尔滨.深圳-临沂-哈尔滨 对应航班及时刻: 1.杭州-郑州航线去程航班号为GJ8767,7:00从杭州起飞,8:1 ...

  8. Yahoo前端优化十四条军规

    相信互联网已经越来越成为人们生活中不可或缺的一部分.Ajax,flex等等富客户端的应用使得人们越加"幸福"地体验着许多原先只能在C/S实 现的功能. 比如Google机会已经把最 ...

  9. 假如有Thread1、Thread2、ThreaD3、Thread4四条线程分别统计C、D、E、F四个盘的大小,所有线程都统计完毕交给Thread5线程去做汇总,应当如何实现?

    一般情况,我们实现多线程都是Thread或者Runnable(后者比较多),但是,这两种都是没返回值的,所以我们需要使用callable(有返回值的多线程)和future(获得线程的返回值)来实现了. ...

最新文章

  1. 2016计算机二级java_2016计算机二级JAVA练习题及答案
  2. linux下tmp目录属性,Linux:文件夹属性及umask
  3. Matplotlib学习---用matplotlib画箱线图(boxplot)
  4. Sklearn 与 TensorFlow 机器学习实用指南(补档)
  5. linux开源开发板 软硬件资料,LeMaker Guitar开源开发板安装系统镜像、Linux 系统快速指南 | 乐美客开源开发板社区...
  6. visual studio 使用快捷方法2
  7. 如何在电脑上搭建一个私服,实现maven项目的上传和下载?
  8. KingTable 是表格动态列插件
  9. Linux下添加删除查看用户操作
  10. 对象取值操作Object.values()
  11. [2018.10.18 T3] 玩串
  12. 站在商业、技术与人文三叉路口的实体书店
  13. iOS 图片编辑——涂鸦——在图片上添加文字
  14. 3D结构光摄像头深度算法
  15. 介值定理究竟在讲什么?
  16. 【PAT B1015】德才论 (c语言)//答案正确
  17. 51单片机接上拉电阻原理
  18. 八皇后问题。。。。。。
  19. 服务器硬盘出现坏道导致数据丢失的数据恢复过程
  20. 英语口语266之每日十句口语

热门文章

  1. mysql 连接超时 wait_timeout interactive_timeout 简介
  2. 内网渗透测试:隐藏通讯隧道技术(上)
  3. c/c++ 避免重复包含 pragma once 和 #ifndef 的区别
  4. linux tree命令 树结构显示目录文件
  5. 使用IRP进行文件操作
  6. tcp connection setup的实现
  7. 批处理 探测在线计算机,批处理(bat)用来监测Windows网络状态脚本
  8. Android 利用方向传感器实现 指南针
  9. 最清楚的mmap()详解与源码分析
  10. thinkphp读写mysql的枷锁_thinkphp5 数据库配置读写分离