【Java】使用 BeanUtils.copyProperties 11个坑(注意事项)

目录

背景

坑1:类型不匹配

坑2:属性名称不一致

坑3:BeanUtils.copyProperties 是浅拷贝

坑4:Null 值覆盖

坑5:注意引入的包

坑6:Boolean 类型数据 + is 开头属性的坑

坑7:查不到字段引用

坑8:不同内部类,即使相同属性,也是赋值失败

坑9:bean 对应的属性,没有 getter 和 setter 方法,复制失败

坑10:BeanUtils.copyProperties + 泛型

坑11:性能问题

替换 BeanUtils.copyProperties 的方案


背景

我们日常开发中,经常涉及到 DODTOVO 对象属性拷贝赋值,很容易想到 org.springframework.beans.BeanUtils 的 copyProperties。它会自动通过反射机制获取源对象和目标对象的属性,并将对应的属性进行复制。可以减少手动编写属性复制代码的工作量,提高代码的可读性和维护性。

 但是你知道嘛?使用 BeanUtils 的 copyProperties,会有好几个坑,下面盘点一下:

坑1:类型不匹配

@Data
public class SourceBean {private Long age;
}@Data
public class TargetBean {private String age;
}public class Test {public static void main(String[] args) {SourceBean source = new SourceBean();source.setAge(25L);TargetBean target = new TargetBean();BeanUtils.copyProperties(source, target);System.out.println(target.getAge());  //拷贝赋值失败,输出null}
}

在上述 demo 中,源对象 SourceBean 的 age 属性是一个 Long 类型,而目标对象 TargetBean 的 age 属性是一个 String 类型。由于类型不匹配,BeanUtils.copyProperties 不会赋值成功的。

坑2:属性名称不一致

public class SourceBean {private String username;// getter 和 setter 方法省略
}public class TargetBean {private String userName;// getter 和 setter 方法省略
}SourceBean source = new SourceBean();source.setUsername("测试");TargetBean target = new TargetBean();BeanUtils.copyProperties(source, target);System.out.println(target.getUserName());   // 输出为 null

在上述示例中,源对象 SourceBean 的属性名称是 username,而目标对象 TargetBean 的属性名称是 userName。但是,两个 username,一个 N 是大写,一个 n 是小写,即属性名称不一致,BeanUtils.copyProperties 方法无法自动映射这些属性(无法忽略大小写自动匹配),因此目标对象的 userName 属性值为 null。

大家在日常开发中,要注意这个坑哈~比如大小写不一致,差一两个字母等等。

坑3:BeanUtils.copyProperties 是浅拷贝

先给大家复习一下,什么是浅拷贝?什么是深拷贝?

浅拷贝是指创建一个新对象,该对象的属性值与原始对象相同,但对于引用类型的属性,仍然共享相同的引用。换句话说,浅拷贝只复制对象及其引用,而不复制引用指向的对象本身。

深拷贝是指创建一个新对象,该对象的属性值与原始对象相同,包括引用类型的属性。深拷贝会递归复制引用对象,创建全新的对象,以确保拷贝后的对象与原始对象完全独立。

public class Address {private String city;//getter 和 setter 方法省略
}public class Person {private String name;private Address address;//getter 和 setter 方法省略
}Person sourcePerson = new Person();sourcePerson.setName("John");Address address = new Address();address.setCity("New York");sourcePerson.setAddress(address);Person targetPerson = new Person();BeanUtils.copyProperties(sourcePerson, targetPerson);sourcePerson.getAddress().setCity("London");System.out.println(targetPerson.getAddress().getCity());  // 输出为 "London"

在上述示例中,源对象 Person 的属性 address 是一个引用类型。当使用 BeanUtils.copyProperties 方法进行属性复制时,实际上只复制了引用,即目标对象 targetPerson 的 address 属性引用和源对象 sourcePerson 的 address 属性引用指向同一个对象。因此,当修改源对象的 address 对象时,目标对象的 address 对象也会被修改。

大家日常开发中,要注意这个坑~

坑4:Null 值覆盖

@Data
public class SourceBean {private String name;private String address;}@Data
public class TargetBean {private String name;private String address;
}SourceBean source = new SourceBean();
source.setName("John");
source.setAddress(null);TargetBean target = new TargetBean();
target.setAddress("address");
BeanUtils.copyProperties(source, target);System.out.println(target.getAddress());  // 输出为 null

在示例中,源对象 SourceBean 的 address 属性值为 null。默认情况下,BeanUtils.copyProperties 方法会将源对象中的 null 值属性覆盖到目标对象中。因此,目标对象的 address 属性值也为 null。

如果你不希望 null 值覆盖目标对象中的属性,可以使用 BeanUtils.copyProperties 方法的重载方法,并传入一个自定义的 ConvertUtilsBean 实例来进行配置。或者使用 Hutool 中的 BeanUtil.copyProperties 方法:

BeanUtil.copyProperties(source, target, ignoreNullValue);

其中,source 是源对象,target 是目标对象,ignoreNullValue 是一个布尔值,表示是否忽略源对象中值为 null 的属性。
 

坑5:注意引入的包

BeanUtils.copyProperties 其实有两个包,分别是 spring、apache。大家注意一下哈,这两个包,是有点不一样的

//org.springframework.beans.BeanUtils(源对象在左边,目标对象在右边)
public static void copyProperties(Object source, Object target) throws BeansException //org.apache.commons.beanutils.BeanUtils(源对象在右边,目标对象在左边)
public static void copyProperties(Object dest, Object orig) throws IllegalAccessException, InvocationTargetException

使用的时候千万要注意,仔细看自己引入的是哪个 BeanUtils,写对应参数位置。

坑6:Boolean 类型数据 + is 开头属性的坑

在 SourceBean 和 TargetBean 中的都有个属性 isTianLuo,他们的数据类型保持不变,但是一个为基本类型 boolean,一个为包装类型 Boolean

@Data
public class SourceBean {private boolean isTianLuo;
}@Data
public class TargetBean {private Boolean isTianLuo;
}SourceBean source = new SourceBean();
source.setTianLuo(true);TargetBean target = new TargetBean();
BeanUtils.copyProperties(source, target);
System.out.println(target.getIsTianLuo()); // 输出为 null

为什么呢?即使一个是包装类型,一个基本类型,应该可以赋值上才对的。

这是因为当属性类型为 boolean 时,属性名以 is 开头,属性名会去掉前面的 is,因此源对象和目标对象属性对不上啦。

大家使用 BeanUtils.copyProperties 过程中,要注意哈~

坑7:查不到字段引用

在某些开发场景中,如果我们要修改某个字段的赋值,我们可能会全文搜索它的所有 set 方法,看有哪些地方引用到。

但是呢,如果使用 BeanUtils.copyProperties,就不知道是否引用到对应的 set 方法啦,即查不到字段引用。这就可能导致会漏掉修改对应字段。

坑8:不同内部类,即使相同属性,也是赋值失败

@Data
public class CopySource {public String outerName;public CopySource.InnerClass innerClass;@Datapublic static class InnerClass {public String InnerName;}
}@Data
public class CopyTarget {public String outerName;public CopyTarget.InnerClass innerClass;@Datapublic static class InnerClass {public String InnerName;}
}CopySource test1 = new CopySource();
test1.outerName = "outTianluo";CopySource.InnerClass innerClass = new CopySource.InnerClass();
innerClass.InnerName = "innerTianLuo";
test1.innerClass = innerClass;System.out.println(test1);
CopyTarget test2 = new CopyTarget();
BeanUtils.copyProperties(test1, test2);System.out.println(test2);  //输出CopyTarget(outerName=outTianluo, innerClass=null)

在 demo 中,CopySource 和 CopyTarget 各自存在一个内部类 InnerClass,虽然这个内部类属性也相同,类名也相同,但 是在不同的类中,因此 Spring 会认为属性不同,不会 copy;若想复制成功,可以让他们指向同一个内部类。

坑9:bean 对应的属性,没有 getter 和 setter 方法,复制失败

BeanUtils.copyProperties 要拷贝属性值成功,需要对应的 bean 要有 getter 和 setter 方法。因为它是用反射拿到 set 和 get 方法,再去拿属性值和设置属性值的。

@Data
public class SourceBean {private String value;
}@Getter   //没有对应的setter方法
public class TargetBean {private String value;
}SourceBean source = new SourceBean();
source.setValue("测试");TargetBean target = new TargetBean();
BeanUtils.copyProperties(source, target);
System.out.println(target.getValue()); //输出null 

坑10:BeanUtils.copyProperties + 泛型

如果 BeanUtils.copyProperties 遇到泛型,也是很可能赋值失败的:

@Data
public class CopySource {public String outerName;public List<CopySource.InnerClass> clazz;@Datapublic static class InnerClass {public String InnerName;}
}@ToString
@Data
public class CopyTarget {public String outerName;public List<CopyTarget.InnerClass> clazz;@Datapublic static class InnerClass {public String InnerName;}
}CopySource test1 = new CopySource();
test1.outerName = "outTianluo";CopySource.InnerClass innerClass = new CopySource.InnerClass();
innerClass.InnerName = "innerTianLuo";List<CopySource.InnerClass> clazz = new ArrayList<>();
clazz.add(innerClass);
test1.setClazz(clazz);System.out.println(test1);
CopyTarget test2 = new CopyTarget();
BeanUtils.copyProperties(test1, test2);System.out.println(test2);  //输出CopyTarget(outerName=outTianluo, clazz=null)

在里面的例子中,BeanUtils.copyProperties 方法拷贝包含泛型属性的对象 clazz。CopyTarget 和 CopySource 的泛型属性类型不匹配,因此拷贝赋值失败。

如果是低版本的包,泛型不匹配会报错,高版本则只是拷贝赋值失败。

坑11:性能问题

由于这些 BeanUtils 类都是采用反射机制实现的,对程序的效率也会有影响:

SourceBean sourceBean = new SourceBean();
sourceBean.setName("tianLuoBoy");
TargetBean target = new TargetBean();long beginTime = System.currentTimeMillis();
for (int i = 0; i < 100000; i++) {  //循环10万次target.setName(sourceBean.getName());
}
System.out.println("common setter time:" + (System.currentTimeMillis() - beginTime));long beginTime1 = System.currentTimeMillis();
for (int i = 0; i < 100000; i++) {  //循环10万次BeanUtils.copyProperties(sourceBean, target);
}
System.out.println("bean copy time:" + (System.currentTimeMillis() - beginTime1));//输出
common setter time:3
bean copy time:331

从上述示例可以发现,简单的 setter 和 BeanUtils.copyProperties 对比,性能差距非常大。因此,慎用 BeanUtils.copyProperties!!!

替换 BeanUtils.copyProperties 的方案

上面聊了 BeanUtils.copyProperties 的11个 坑,都是在告诉大家要慎用 BeanUtils.copyProperties。那么有没有推荐替换它的方案呢?

  • 第一种,使用原始的 setter 和 getter 方法

使用手动的 setter 方法进行属性赋值。这种方法可能需要编写更多的代码,但是可以提供更细粒度的控制,并且在性能方面通常比 BeanUtils.copyProperties 更高效。

Target target = new Target();
target.setName(source.getName());
target.setAge(source.getAge());

如果实在对象 bean 的属性比较多的话,可以使用插件 GenerateAllSetter,它可以一键生成对象的 set 方法,挺方便的。

 

  • 第二种,使用映射工具库

如 MapStruct、ModelMapper等,它们可以自动生成属性映射的代码。这些工具库可以减少手动编写 setter 方法的工作量,并提供更好的性能。

使用 MapStruct 的示例:

@Mapper
public interface SourceTargetMapper {SourceTargetMapper INSTANCE = Mappers.getMapper(SourceTargetMapper.class);@Mapping(source = "name", target = "name")@Mapping(source = "age", target = "age")Target mapToTarget(Source source);
}Target target = SourceTargetMapper.INSTANCE.mapToTarget(source);

好事定律:每件事最后都会是好事,如果不是好事,说明还没到最后。

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

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

相关文章

【Linux】常用基本指令汇总

前言&#xff1a; 本章将介绍Linux操作系统常用的基本指令&#xff0c;另外&#xff0c;使用这些指令编辑一个shell脚本&#xff0c;方便大家理解使用。 目录 常用指令whoamipwdls关于iNode的解释验证标识文件的方式 cdtouchmkdir&#xff08;重要&#xff09;treemdir指令 &a…

Jmeter断言、关联、脚本录制

Jmeter断言 断言&#xff1a;让程序自动判断预期结果和实际结果是否一致 提示: Jmeter在请求的返回层面有个自动判断机制&#xff08;响应状态码 2xx:成功&#xff0c;4xx/5xx:失败&#xff09;但是请求成功了&#xff0c;并不代表结果一定正确&#xff0c;因此需要检测机制…

【保姆级图文教程】QT下载、安装、入门、配置VS Qt环境

【保姆级图文教程】QT下载、安装、入门、配置VS Qt环境-CSDN博客 0.QT介绍 QT 是一个跨平台的应用程序开发框架&#xff0c;它提供了丰富的工具和类库&#xff0c;用于开发图形用户界面&#xff08;GUI&#xff09;程序。Qt 提供了 C 编程语言接口&#xff0c;同时也支持其他…

王学岗鸿蒙开发(北向)——————(一)鸿蒙开发环境的搭建与ArkTs介绍

1&#xff0c;鸿蒙系统开始研发的时间是在2012年。 2&#xff0c;目前鸿蒙有两个开发:HarmonyOS和OpenHarmony,前者内聚AOSP(Android的东西)&#xff0c;前者是双框架结构&#xff0c;后者不是双框架结构&#xff0c;没有内置安卓。 3&#xff0c;Harmony地址 4&#xff0c;我们…

文件上传漏洞之upload-labs

前提&#xff1a; 本文中的以xshell命名的均为以密码为admin的一句话木马&#xff0c;而shell命名的则是由冰蝎工具生成的木马。 pass-01&#xff1a;js前端验证 测试性的上传一个一句话木马&#xff0c;发现被拦截了&#xff0c;而且根据推测大概率是前端检测&#xff0c;于…

国内PLM系统厂商,国内PLM系统哪个公司最受欢迎

国内PLM系统厂商,国内PLM系统哪个公司最受欢迎 国内PLM系统厂商中&#xff0c;要确定哪个公司的产品最受欢迎&#xff0c;需要考虑多个因素&#xff0c;包括市场份额、客户评价、技术实力、产品线完整性以及服务支持等。虽然无法直接给出一个具体的“最受欢迎”的排名&#xff…

闲鱼无货源-高级班,最全·最新·最干,紧贴热点 深度学习(17节课)

课程目录 1-1&#xff1a;闲鱼潜规则_1.mp4 2-2&#xff1a;闲鱼的基础操作-养号篇_1.mp4 3-3&#xff1a;闲鱼实战运营-选品篇&#xff08;一&#xff09;_1.mp4 4-4&#xff1a;闲鱼实战运营-选图视频篇_1.mp4 5-5&#xff1a;闲鱼实战运营-标题筒_1.mp4 6-6&#xff1…

伯克希尔也被ST?

地球&#xff08;最&#xff09;贵股票突然闪崩&#xff0c;美股故障再次上演。昨晚齐刷刷一片的美股出现行情异常&#xff0c;伯克希尔、蒙特利尔银行、巴里克黄金等股票股价跌幅超过98%。其中&#xff0c;巴菲特旗下伯克希尔哈撒韦A类股股价更是暴跌99.97%&#xff0c;股价从…

Java集合简略记录

一、集合体系结构 单列集合&#xff1a;Collection 双列集合&#xff1a;Map 二、单列集合 List系列集合&#xff1a;添加的元素是有序、可重复、有索引 有序指的是存和取的顺序是一致的&#xff0c;和之前排序的从小到大是没有任何关系的 Set系列集合&#xff1a;添加的元素是…

FL Studio21.2.8中文版水果音乐制作的革新之旅!

在数字化浪潮的推动下&#xff0c;音乐制作领域经历了翻天覆地的变化。从最初的模拟技术到如今的全数字化处理&#xff0c;音乐制作的门槛被大幅降低&#xff0c;越来越多的音乐爱好者和专业人士开始尝试自行创作和编辑音乐。在这个过程中&#xff0c;各种专业音乐制作软件成为…

yolov8逐步分解(9)_训练过程之Epoch迭代过程

yolov8逐步分解(1)--默认参数&超参配置文件加载_yolov8 加载yaml配置 预测-CSDN博客 yolov8逐步分解(2)_DetectionTrainer类初始化过程_train and val are required in all data yamls.-CSDN博客 yolov8逐步分解(3)_trainer训练之模型加载_yolov8 加载模型-CSDN博客 YOL…

TCP 建链(三次握手)和断链(四次握手)

TCP 建链&#xff08;三次握手&#xff09;和断链&#xff08;四次挥手&#xff09; 背景简介建链&#xff08;三次握手&#xff09;断链&#xff08;四次挥手&#xff09;序号及标志位延伸问题为什么建立连接需要握手三次&#xff0c;两次行不行&#xff1f;三次握手可以携带数…

智领未来,安全无忧:智能网联汽车监控大屏的守护之旅

在繁忙的都市中&#xff0c;驾驶者往往面临着诸多安全隐患。传统的驾驶辅助系统虽然能够提供一定的帮助&#xff0c;但在复杂多变的交通环境中&#xff0c;其局限性也逐渐显现。而智能网联汽车安全监控大屏&#xff0c;正是为了解决这一问题而诞生的。 山海鲸可视化大屏 大屏采…

基础篇04——多表查询

多表关系 一对多 多对多 多对多是通过中间表实现的 -- 创建学生表 create table student (id int auto_increment primary key comment ID,name varchar(10) comment 姓名,no varchar(3) comment 学号 ) comment 学生表;insert into student values (null, 黛绮丝, 001),(…

SLAM中四元数、流形、李群、李代数是啥?

知识点得逻辑关系如下 引言 非线性问题由于复杂的数学结构&#xff0c;多样的解空间&#xff0c;局部极值等问题求解难度大大增加。所以在求解时需要把非线性问题转化为更容易处理的形式&#xff0c;例如 数值优化方法&#xff1a;数值优化方法包括梯度下降、共轭梯度法、牛顿…

Spring boot实现基于注解的aop面向切面编程

Spring boot实现基于注解的aop面向切面编程 背景 从最开始使用Spring&#xff0c;AOP和IOC的理念就深入我心。正好&#xff0c;我需要写一个基于注解的AOP&#xff0c;被这个注解修饰的参数和属性&#xff0c;就会被拿到参数并校验参数。 一&#xff0c;引入依赖 当前sprin…

寒武纪:“国产平替”道阻且长

英伟达的一季报反映了AI装备竞赛白热化&#xff0c;科技巨头们正疯狂加码投资。 不过&#xff0c;因为众所周知的原因,英伟达最先进的产品卖不到国内&#xff0c;虽然说一定程度上限制了咱们AI的发展&#xff0c;但也给国产AI芯片公司们提供了机会&#xff0c;其中就包括我们今…

K8S==ingress简单搭建和使用

基础环境 D:\DOCKER_REPO\K8S>kubectl version Client Version: v1.29.2 Kustomize Version: v5.0.4-0.20230601165947-6ce0bf390ce3 Server Version: v1.29.2 D:\DOCKER_REPO\K8S>kubectl get nodes NAME STATUS ROLES AGE VERSION docker-…

01_深度学习基础知识

1. 感知机 感知机通常情况下指单层的人工神经网络,其结构与 MP 模型类似(按照生物神经元的结构和工作原理造出来的一个抽象和简化了模型,也称为神经网络的一个处理单元) 假设由一个 n 维的单层感知机,则: x 1 x_1 x1​ 至 x n x_n xn​ 为 n 维输入向量的各个分量w 1 j…

【大学物理】Interference,diffraction,polarization:光学

nature of light definition speed of light reflection dispersion huygenss principle:惠更斯原理