基于Beanutils.copyProperties()的用法及重写提高效率
作者:菜鸟凯一枚 发布时间:2023-04-20 12:10:25
Beanutils.copyProperties()用法及重写提高效率
特别说明本文介绍的是Spring(import org.springframework.beans.BeanUtils)中的BeanUtils.copyProperties(A,B)方法。是将A中的值赋给B。apache(org.apache.commons.beanutils.BeanUtils)中的BeanUtils.copyProperties(A,B)方法是将B中的值赋值给A。
一、简介
BeanUtils提供对Java反射和自省API的包装。其主要目的是利用反射机制对JavaBean的属性进行处理。我们知道,一个JavaBean通常包含了大量的属性,很多情况下,对JavaBean的处理导致大量get/set代码堆积,增加了代码长度和阅读代码的难度。
二、用法
如果你有两个具有很多相同属性的JavaBean,一个很常见的情况就是Struts里的PO对象(持久对象)和对应的ActionForm。例如:一个用户注册页面,有一个User实体类和一个UserActionForm,我们一般会在Action里从ActionForm构造一个PO对象,传统的方式是使用类似下面的语句对属性逐个赋值:
// 获取 ActionForm 表单数据
UserActionForm uForm = (UserActionForm) form;
// 构造一个User对象
User user = new User();
// 逐一赋值
user.setUsername(uForm.getUsername);
user.setPassword(uForm.getPassword);
user.setAge(uForm.getAge);
...........
...........
// 然后调用JDBC、或操作Hibernate 持久化对象User到数据库
HibernateDAO.save(user);
通过这样的方法如果表单数据N多、100、1000(夸张点。哈哈)、、、、那我们不是要写100、、、1000行set、get了。谁都
不愿意这样做。
而我们使用 BeanUtils.copyProperties() 方法以后,代码量大大的减少,而且整体程序看着也简洁明朗,代码如下:
// 获取 ActionForm 表单数据
UserActionForm uForm = (UserActionForm) form;
// 构造一个User对象
User user = new User();
BeanUtils.copyProperties(uForm,user);
// 然后调用JDBC、或操作Hibernate 持久化对象User到数据库
HibernateDAO.save(user);
注:如果User和UserActionForm 间存在名称不相同的属性,则BeanUtils不对这些属性进行处理,需要手动处理。例如:
User类里面有个createDate 创建时间字段,而UserActionForm里面无此字段。BeanUtils.copyProperties()不会对此字段做任何处理。必须要自己手动处理。
user.setModifyDate(new Date());
三、重写
ReflectASM,高性能的反射:
什么是ReflectASM ReflectASM是一个很小的java类库,主要是通过asm生产类来实现java反射,执行速度非常快,看了网上很多和反射的对比,觉得ReflectASM比较神奇,很想知道其原理,下面介绍下如何使用及原理;
public static void main(String[] args) {
User user = new User();
//使用reflectasm生产User访问类
MethodAccess access = MethodAccess.get(User.class);
//invoke setName方法name值
access.invoke(user, "setName", "张三");
//invoke getName方法 获得值
String name = (String)access.invoke(user, "getName", null);
System.out.println(name);
}
原理
上面代码的确实现反射的功能,代码主要的核心是 MethodAccess.get(User.class);
看了下源码,这段代码主要是通过asm生产一个User的处理类 UserMethodAccess(这个类主要是实现了invoke方法)的ByteCode,然后获得该对象,通过上面的invoke操作user类。
private static Map<Class, MethodAccess> methodMap = new HashMap<Class, MethodAccess>();
private static Map<String, Integer> methodIndexMap = new HashMap<String, Integer>();
private static Map<Class, List<String>> fieldMap = new HashMap<Class, List<String>>();
public static void copyProperties(Object desc, Object orgi) {
MethodAccess descMethodAccess = methodMap.get(desc.getClass());
if (descMethodAccess == null) {
descMethodAccess = cache(desc);
}
MethodAccess orgiMethodAccess = methodMap.get(orgi.getClass());
if (orgiMethodAccess == null) {
orgiMethodAccess = cache(orgi);
}
List<String> fieldList = fieldMap.get(orgi.getClass());
for (String field : fieldList) {
String getKey = orgi.getClass().getName() + "." + "get" + field;
String setkey = desc.getClass().getName() + "." + "set" + field;
Integer setIndex = methodIndexMap.get(setkey);
if (setIndex != null) {
int getIndex = methodIndexMap.get(getKey);
// 参数一需要反射的对象
// 参数二class.getDeclaredMethods 对应方法的index
// 参数对三象集合
descMethodAccess.invoke(desc, setIndex.intValue(),
orgiMethodAccess.invoke(orgi, getIndex));
}
}
}
// 单例模式
private static MethodAccess cache(Object orgi) {
synchronized (orgi.getClass()) {
MethodAccess methodAccess = MethodAccess.get(orgi.getClass());
Field[] fields = orgi.getClass().getDeclaredFields();
List<String> fieldList = new ArrayList<String>(fields.length);
for (Field field : fields) {
if (Modifier.isPrivate(field.getModifiers())
&& !Modifier.isStatic(field.getModifiers())) { // 是否是私有的,是否是静态的
// 非公共私有变量
String fieldName = StringUtils.capitalize(field.getName()); // 获取属性名称
int getIndex = methodAccess.getIndex("get" + fieldName); // 获取get方法的下标
int setIndex = methodAccess.getIndex("set" + fieldName); // 获取set方法的下标
methodIndexMap.put(orgi.getClass().getName() + "." + "get"
+ fieldName, getIndex); // 将类名get方法名,方法下标注册到map中
methodIndexMap.put(orgi.getClass().getName() + "." + "set"
+ fieldName, setIndex); // 将类名set方法名,方法下标注册到map中
fieldList.add(fieldName); // 将属性名称放入集合里
}
}
fieldMap.put(orgi.getClass(), fieldList); // 将类名,属性名称注册到map中
methodMap.put(orgi.getClass(), methodAccess);
return methodAccess;
}
}
执行1000000条效率80几毫秒,效率已经没问题了。
BeanUtils.copyProperties 使用注意
首先结论说在前头, BeanUtils.copyProperties 是浅拷贝 。
为什么今天我还想把这个BeanUtils.copyProperties 的使用拿出来军训。
因为我意识到了大家(部分)对深拷浅拷还是不清晰,不知道具体的影响。
示例演示
第一个类:
第二个类:
注意!! 第二个类里面有使用第一个类。
开始进行示例
public static void main(String[] args) {
/**
* 模拟数据 A complexObject
*/
ComplexObject complexObjectA=new ComplexObject();
complexObjectA.setNickName("张一");
SimpleObject simpleObject=new SimpleObject();
simpleObject.setName("李四");
simpleObject.setAge(12);
complexObjectA.setSimpleObject(simpleObject);
/**
* 使用BeanUtils.copyProperties 拷贝 模拟数据 A 生成模拟数据 B
*/
ComplexObject complexObjectB=new ComplexObject();
BeanUtils.copyProperties(complexObjectA,complexObjectB);
System.out.println("拷贝后,查看模拟数据A 和 模拟数据B :");
System.out.println(complexObjectA.getSimpleObject().toString());
System.out.println(complexObjectB.getSimpleObject().toString());
System.out.println("比较模拟数据A 和 模拟数据B 里面的引用对象simple 是否引用地址一样: ");
System.out.println(complexObjectA.getSimpleObject()==complexObjectB.getSimpleObject());
System.out.println("修改拷贝出来的模拟数据B里面的引用对象simple的属性 age 为 888888");
complexObjectB.getSimpleObject().setAge(888888);
System.out.println("修改后,观察原数据A 和拷贝出来的数据 B 里面引用的 对象 simple的属性 age:");
System.out.println(complexObjectA.getSimpleObject().toString());
System.out.println(complexObjectB.getSimpleObject().toString());
}
最后强调
如果你是使用BeanUtils.copyProperties 进行对象的拷贝复制, 一定要注意!
第一点 、你所拷贝的对象内包不包含 其他对象的引用。
第二点、如果包含,那么接下来的方法里无论是操作原对象还是操作拷贝出来的对象是否涉及到 对 对象内 的 那个其他对象的 值的修改 。
第三点、如果涉及到, 修改了,会不会影响到其他方法 对 修改值的使用情况。
就如文中例子, 如果传入过来的一个复杂对象数据A 里面引用了一个 user对象年龄age是10;拷贝出一份数据B后, 操作 数据B的方法把 年龄age改成了88888;
那么后续其他方法用到数据A ,想用的是最初始的 age 为10 ,那么就用不到了,因为浅拷贝的原因受影响,age都变成88888 了。
ok,就提到这。以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家~
来源:https://blog.csdn.net/qq_37782076/article/details/86605282


猜你喜欢
- MQ:消息队列/消息中间件/消息代理,产品有很多,ActiveMQ RabbitMQ RocketMQ Kafka1、Stream解决的痛点
- 本文实例为大家分享了C#实现飞行棋的具体代码,供大家参考,具体内容如下游戏规则如果玩家A踩到了玩家B,玩家B退6格踩到了1幸运轮盘,a交换位
- Java代码 mvn install:install-file -DgroupId=包名 -DartifactId=项目名 -Dversio
- 牛逼!IDEA不愧为神器,结合Groovy脚本,简直天下无敌,如今, 有许许多多的插件或者编辑器都支持根据数据表自动生成数据实体类了, 比如
- 线程死锁是指由于两个或者多个线程互相持有对方所需要的资源,导致这些线程处于等待状态,无法前往执行。当线程进入对象的synchronized代
- 首先,将json串转为一个JObject对象:JObject jo = (JObject)JsonConvert.DeserializeOb
- 符号 ASCII码 &
- [LeetCode] 2. Add Two Numbers 两个数字相加You are given two non-empty&n
- 知识点回顾封装封装(有时称为数据隐藏)是与对象有关的一个重要概念。从形式上来看,封装不过是将数据和行为组合在一个包中,并对对象的使用者隐藏了
- 简介对于一个APP来说,肯定会有一个AppBar,这个AppBar一般包含了APP的导航信息等。虽然我们可以用一个固定的组件来做为AppBa
- 一、安装MongoDB4.0.3(××)1.1、官方安装文档https://docs.mongodb.com/manual/tutorial
- 1.为什么需要动态内存分配关于这个问题,我们先看看我们之前是如何开辟内存的。int val = 20;//在栈空间上开辟四个字节char a
- 在实际应用中,很可能我们希望自己的app在按下返回键的时候并不退出,而是像按home键一样仅仅返回桌面,而程序仍然在后台运行着。要怎么实现这
- Android中实现定时器的四种方式第一种方式利用Timer和TimerTask1、继承关系java.util.Timer基本方法sched
- 目录卡顿原理卡顿监控ANR原理卡顿原理主线程有耗时操作会导致卡顿,卡顿超过阀值,触发ANR。 应用进程启动时候,Zygote会反射调用Act
- 本文实例为大家分享了Unity实现俄罗斯方块第一部分,供大家参考,具体内容如下准备工作1、新建一个2D项目,新建成功以后设置相机的一些参数2
- Android 读取文件内容实现方法,这里整理了几种方法,大家需要可以看下。如果要打开存放在/data/data/<package n
- SpringBoot打jar包遇到的xml文件丢失在pom.xml的build标签中添加如下内容指定资源路径<resources>
- @Conditional的使用@Conditional可以根据条件来判断是否注入某些Bean。package com.morris.spri
- 使用线程池的好处1、降低资源消耗可以重复利用已创建的线程降低线程创建和销毁造成的消耗。2、提高响应速度当任务到达时,任务可以不需要等到线程创