解决BeanUtils.copyProperties不支持复制集合的问题

工作中,经常使用Spring的工具类BeanUtils.copyProperties对bean属性进行复制,这里的复制属于浅复制。且不能复制集合和数组。本文会对该工具进行一些测试。

文末会提出复制集合属性的解决方案

准备工作:准备测试需要的类

@Datapublic class Class {    private People[] member;    private People teacher;    private List<People> student;}@Data@NoArgsConstructor@AllArgsConstructorpublic 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));}

测试结果

是否可以复制数组:falseBeanUtils.copyProperties是否可以复制List:false通过JSON工具复制List:[People(id=2, name=班长, age=15, sex=1), People(id=3, name=学生, age=15, sex=1)]通过JSON工具是否深复制:trueBeanUtils.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,

测试数据长度:1JSON:154767336Bean:275182853测试数据长度:10JSON:165678435Bean:275301421测试数据长度:100JSON:167937206Bean:328461161测试数据长度:1000JSON:187832969Bean:315815289测试数据长度:10000JSON:297461312Bean:362763360测试数据长度:100000JSON:562035707Bean: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方法),以及通过串行化将其返回到远程的客户机的时间总和。所以要小心使用这种威力!

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

旅行,其实是需要具有一些流浪精神的,

解决BeanUtils.copyProperties不支持复制集合的问题

相关文章:

你感兴趣的文章:

标签云: