BeanUtils.copyProperties使用总结以及注意事项说明

目录

  • 1.前言
  • 2.一般使用
  • 3.拷贝属性时忽略空值
  • 4.使用注意事项(1)
  • 5.使用注意事项(2)
  • 6.使用注意事项(3)

1.前言
开发过程中,讲一个对象的属性和值赋值到另一个对象上,大量使用了get、set方法,看着很臃肿,思考下肯定不只有我有这种想法,所以技术上肯定有方法能解决这个问题,所以查阅了一些资料发现了BeanUtils.copyProperties这个方法以下是这次所有的总结以及使用时的注意事项。
使用org.springframework.beans.BeanUtils.copyProperties方法进行对象之间属性的赋值,避免通过get、set方法一个一个属性的赋值。

2.一般使用
BeanUtils是这个包里比较常用的一个工具类,该方法定义如下:
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对象,传统的方式是使用类似下面的语句对属性逐 个赋值:
//1得到TeacherFormTeacherForm teacherForm=(TeacherForm)form; //2构造Teacher对象Teacher teacher=new Teacher(); //3赋值teacher.setName(teacherForm.getName()); teacher.setAge(teacherForm.getAge()); teacher.setGender(teacherForm.getGender()); teacher.setMajor(teacherForm.getMajor()); teacher.setDepartment(teacherForm.getDepartment()); //4持久化Teacher对象到数据库HibernateDAO=; HibernateDAO.save(teacher);

而使用BeanUtils后,代码就大大改观了,如下所示:
//1得到TeacherFormTeacherForm teacherForm=(TeacherForm)form; //2构造Teacher对象Teacher teacher=new Teacher(); //3赋值BeanUtils.copyProperties(teacher,teacherForm); //4持久化Teacher对象到数据库HibernateDAO=; HibernateDAO.save(teacher);

如 果Teacher和TeacherForm间存在名称不相同的属性,则BeanUtils不对这些属性进行处理,需要程序员手动处理。
例如 Teacher包含modifyDate(该属性记录最后修改日期,不需要用户在界面中输入)属性而TeacherForm无此属性,那么在上面代码的 copyProperties()后还要加上一句:
teacher.setModifyDate(new Date());


3.拷贝属性时忽略空值
使用BeanUtils.copyProperties有一个问题就是当src对象的键值为Null时
就会把target对象的对应键值覆盖成空了,这明显不是我们想要的,以下这个方法可以解决
/** * 复制属性,过滤掉不复制的属性 */public static void copyBeanProperties(final Object source,//1,待复制的原始对象final Object target,//2,复制后的结果对象//3,获取保存你不需要复制的属性名final Collection excludes = new ArrayList(); final PropertyDescriptor[] propertyDescriptors = BeanUtils.getPropertyDescriptors(source.getClass()); for(final PropertyDescriptor propertyDescriptor : propertyDescriptors){String propName = propertyDescriptor.getName(); if(!includes.contains(propName)){excludes.add(propName); }}//4,复制操作BeanUtils.copyProperties(source, target, excludes.toArray(new String[excludes.size()])); }

使用案例
public static String[] getNullPropertyNames (Object source) {final BeanWrapper src = https://www.it610.com/article/new BeanWrapperImpl(source); java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors(); Set emptyNames = new HashSet(); for(java.beans.PropertyDescriptor pd : pds) {Object srcValue = src.getPropertyValue(pd.getName()); if (srcValue == null) emptyNames.add(pd.getName()); }String[] result = new String[emptyNames.size()]; return emptyNames.toArray(result); }public static void copyPropertiesIgnoreNull(Object src, Object target){BeanUtils.copyProperties(src, target, getNullPropertyNames(src)); }


4.使用注意事项(1)
Property.copyProperties()和BeanUtils.copyProperties()
除BeanUtils外还有一个名为PropertyUtils的工具类,它也提供copyProperties()方法
  • 1.无论是org.springframework.beans或者org.apache.commons.beanutils,与get/set方式相比,都存在性能问题。
  • 2.效率由高到底:get/set 》PropertyUtils 》BeanUtils。
  • 3.PropertyUtils和BeanUtils两个工具类都是对bean之间存在属性名相同的属性进行处理,无论是源bean或者是目标bean中多出来的属性均不处理。
  • 4.具体来说:BeanUtils.copyProperties()可以在一定范围内进行类型转换,同时还要注意一些不能转换时候,会将默认null值转化成0; Property.copyProperties()则是严格的类型转化,必须类型和属性名完全一致才转化。对于null的处理:PropertyUtils支持为null的场景;BeanUtils对部分属性不支持null,具体如下:
    • a. java.util.Date类型不支持,但是它的自雷java.sql.Date是被支持的。java.util.Date直接copy会报异常;
    • b. Boolean,Integer,Long等不支持,会将null转化为0;
    • c. String支持,转化后依然为null。
  • 5.BeanUtils的高级功能org.apache.commons.beanutils.Converter接口可以自定义类型转化,也可以对部分类型数据的null值进行特殊处理,如ConvertUtils.register(new DateConverter(null), java.util.Date.class); 但是PropertyUtils没有。
另外:值得注意的是,在测试过程中发现,commons-beanutils-1.8.0.jar版本中的BeanUtils类,支持Byte到Integer或int的转化。说明实际使用过程中,我们还是要多看源码,多做测试,并且注意版本号升级带来的微小变化。
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异常。

5.使用注意事项(2)
在Java中copyProperties() 这个方法出处有两个地方,
BeanUtils是org.springframework.beans.BeanUtils
BeanUtils是org.apache.commons.beanutils.BeanUtils
下面具体说说他们的用法和区别。这个方法在不同的包下面,而这两个类的copyProperties()方法里面传递的参数赋值是相反的。
例如:a,b为对象 BeanUtils.copyProperties(a, b);
BeanUtils是org.springframework.beans.BeanUtils//a拷贝到b//a1 源文件,b1 目标文件public static void copyProperties(Object a1, Object b1)throws BeansException{copyProperties(a1, b1, null, (String[])null); }

BeanUtils是org.apache.commons.beanutils.BeanUtils
//b拷贝到a//a2目标文件 ,b2 原始的,源文件public static void copyProperties(Object a2, Object b2)throws IllegalAccessException, InvocationTargetException{BeanUtilsBean.getInstance().copyProperties(a2, b2); }

引用包出处不一样,意思就不一样,使用的时候一定要看清楚是哪个包下面的。

6.使用注意事项(3)
方法使用起来是不是即省事又舒服?but。。。get/set写代码省事是要付出代价的,那就是使用BeanUtils的运行成本也惊人!BeanUtils所花费的时间要超过取数 据、将其复制到对应的 value对象(通过手动调用get和set方法),以及通过串行化将其返回到远程的客户机的时间总和。所以要小心使用这种威力!有失必有得,反之亦然。
【BeanUtils.copyProperties使用总结以及注意事项说明】以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

    推荐阅读