解决BeanUtils.copyProperties不支持复制集合的问题
工作中,经常使用Spring的工具类BeanUtils.copyProperties对bean属性进行复制,这里的复制属于浅复制。且不能复制集合和数组。本文会对该工具进行一些测试。
文末会提出复制集合属性的解决方案
准备工作:准备测试需要的类
@Data public class Class { private People[] member; private People teacher; private List<People> student; } @Data @NoArgsConstructor @AllArgsConstructor public class People { private Integer id; private String name; private Integer age; private Integer sex; }
测试代码:测试BeanUtils.copyProperties是否支持复制数组和集合,还有解决方案
public static void main(String[] args) { // 测试数组的复制 People[] member = new People[3]; member[0] = new People(1, "老师", 30, 1); member[1] = new People(2, "班长", 15, 1); member[2] = new People(3, "学生", 15, 1); People[] member1 = new People[]{}; BeanUtils.copyProperties(member, member1); System.out.println("是否可以复制数组:" + (member1.length == 0 ? false : true)); // 测试List的复制(Map也不能复制,测试略) List<People> student = new ArrayList<>(); student.add(member[1]); student.add(member[2]); List<People> student1 = new ArrayList<>(); BeanUtils.copyProperties(student, student1); System.out.println("BeanUtils.copyProperties是否可以复制List:" + (student1.isEmpty() ? false : true)); // 通过JSON工具实现List的复制(不仅仅是List,数组和Map等也可以通过类似方法实现复制,需要有无参构造方法,否则报错) student1 = JSON.parseArray(JSON.toJSONString(student), People.class); System.out.println("通过JSON工具复制List:" + student1); System.out.println("通过JSON工具是否深复制:" + (student.get(0) != student1.get(0) ? true : false)); // 测试是否深复制 Class source = new Class(); source.setMember(member); source.setTeacher(member[0]); source.setStudent(student); Class target = new Class(); BeanUtils.copyProperties(source, target); System.out.println("BeanUtils.copyProperties是否深复制:" + (source.getMember() != target.getMember() ? true : false)); }
测试结果
是否可以复制数组:false
BeanUtils.copyProperties是否可以复制List:false
通过JSON工具复制List:[People(id=2, name=班长, age=15, sex=1), People(id=3, name=学生, age=15, sex=1)]
通过JSON工具是否深复制:true
BeanUtils.copyProperties是否深复制:false
针对List的复制除了通过JSON工具,最简单的就是循环复制集合属性,下面测试两种方法的效率。
public static void main(String[] args) { int count = 1; System.out.println("测试数据长度:" + count); List<People> source = new LinkedList<>(); List<People> target = new LinkedList<>(); long start; for (int i = 0; i < count; i++) { source.add(new People(1, "ly", 25, 1)); } start = System.nanoTime(); target = JSON.parseArray(JSON.toJSONString(source), People.class); System.out.println("JSON:" + (System.nanoTime() - start)); start = System.nanoTime(); for (int i = 0; i < count; i++) { People p = new People(); BeanUtils.copyProperties(source.get(i), p); target.add(p); } System.out.println("BeanUtils.copyProperties" + (System.nanoTime() - start)); }
分别测试count=1、10、100、1000、10000、100000的结果。为了防止一起执行出现影响,每次只测试一种复制方法的一种情况,共执行12次。通过对比可以知道,通过JSON复制属性快于BeanUtils,
测试数据长度:1
JSON:154767336
Bean:275182853
测试数据长度:10
JSON:165678435
Bean:275301421
测试数据长度:100
JSON:167937206
Bean:328461161
测试数据长度:1000
JSON:187832969
Bean:315815289
测试数据长度:10000
JSON:297461312
Bean:362763360
测试数据长度:100000
JSON:562035707
Bean:5815319343
通过以下方式解决复制List、Map
public static <T> List copyList(List<T> list) { if (CollectionUtils.isEmpty(list)) { return new ArrayList(); } return JSON.parseArray(JSON.toJSONString(list), list.get(0).getClass()); } public static Map<String, Object> copyMap(Map map) { return JSON.parseObject(JSON.toJSONString(map)); }
BeanUtils.copyProperties的用法和优缺点
一、简介
BeanUtils提供对Java反射和自省API的包装。其主要目的是利用反射机制对JavaBean的属性进行处理。我们知道,一个JavaBean通常包含了大量的属性,很多情况下,对JavaBean的处理导致大量get/set代码堆积,增加了代码长度和阅读代码的难度。
二、用法
BeanUtils是这个包里比较常用的一个工具类,这里只介绍它的copyProperties()方法。该方法定义如下:
Java代码
public static void copyProperties(java.lang.Object dest,java.lang.Object orig) throws java.lang.IllegalAccessException, java.lang.reflect.InvocationTargetException
如果你有两个具有很多相同属性的JavaBean,一个很常见的情况就是Struts里的PO对象(持久对象)和对应的ActionForm,例如 Teacher和TeacherForm。
我们一般会在Action里从ActionForm构造一个PO对象,传统的方式是使用类似下面的语句对属性逐个赋值:
//得到TeacherForm TeacherForm teacherForm=(TeacherForm)form; //构造Teacher对象 Teacher teacher=new Teacher(); //赋值 teacher.setName(teacherForm.getName()); teacher.setAge(teacherForm.getAge()); teacher.setGender(teacherForm.getGender()); teacher.setMajor(teacherForm.getMajor()); teacher.setDepartment(teacherForm.getDepartment()); //持久化Teacher对象到数据库 HibernateDAO.save(teacher);
而使用BeanUtils后,代码就大大改观了,如下所示:
//得到TeacherForm TeacherForm teacherForm=(TeacherForm)form; //构造Teacher对象 Teacher teacher=new Teacher(); //赋值 BeanUtils.copyProperties(teacher,teacherForm); //持久化Teacher对象到数据库 HibernateDAO.save(teacher);
如果Teacher和TeacherForm间存在名称不相同的属性,则BeanUtils不对这些属性进行处理,需要程序员手动处理。
例如 Teacher包含modifyDate(该属性记录最后修改日期,不需要用户在界面中输入)属性而TeacherForm无此属性,那么在上面代码的 copyProperties()后还要加上一句:
teacher.setModifyDate(new Date());
怎么样,很方便吧!除BeanUtils外还有一个名为PropertyUtils的工具类,它也提供copyProperties()方法,作用与 BeanUtils的同名方法十分相似,主要的区别在于后者提供类型转换功能,即发现两个JavaBean的同名属性为不同类型时,在支持的数据类型范围内进行转换,而前者不支持这个功能,但是速度会更快一些。
BeanUtils支持的转换类型如下:
* java.lang.BigDecimal * java.lang.BigInteger * boolean and java.lang.Boolean * byte and java.lang.Byte * char and java.lang.Character * java.lang.Class * double and java.lang.Double * float and java.lang.Float * int and java.lang.Integer * long and java.lang.Long * short and java.lang.Short * java.lang.String * java.sql.Date * java.sql.Time * java.sql.Timestamp
这里要注意一点,java.util.Date是不被支持的,而它的子类java.sql.Date是被支持的。因此如果对象包含时间类型的属性,且希望被转换的时候,一定要使用java.sql.Date类型。否则在转换时会提示argument mistype异常。
三、优缺点
Apache Jakarta Commons项目非常有用。我曾在许多不同的项目上或直接或间接地使用各种流行的commons组件。其中的一个强大的组件就是BeanUtils。我 将说明如何使用BeanUtils将local实体bean转换为对应的value 对象:
BeanUtils.copyProperties(aValue, aLocal)
上面的代码从aLocal对象复制属性到aValue对象。它相当简单!它不管local(或对应的value)对象有多少个属性,只管进行复制。我们假设 local对象有100个属性。
上面的代码使我们可以无需键入至少100行的冗长、容易出错和反复的get和set方法调用。这太棒了!太强大了!太有用 了!
现在,还有一个坏消息:使用BeanUtils的成本惊人地昂贵!我做了一个简单的测试,BeanUtils所花费的时间要超过取数 据、将其复制到对应的 value对象(通过手动调用get和set方法),以及通过串行化将其返回到远程的客户机的时间总和。所以要小心使用这种威力!
以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。