1.前言
开发过程中,讲一个对象的属性和值赋值到另一个对象上,大量使用了get、set方法,看着很臃肿,思考下肯定不只有我有这种想法,所以技术上肯定有方法能解决这个问题,所以查阅了一些资料发现了BeanUtils.copyProperties这个方法以下是这次所有的总结以及使用时的注意事项。
使用org.springframework.beans.BeanUtils.copyProperties方法进行对象之间属性的赋值,避免通过get、set方法一个一个属性的赋值。
2.一般使用
BeanUtils是这个包里比较常用的一个工具类,该方法定义如下:
1
2
3
|
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
2
3
4
5
6
7
8
9
10
11
12
13
|
//1得到TeacherForm
TeacherForm 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
2
3
4
5
6
7
8
9
|
//1得到TeacherForm
TeacherForm 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()后还要加上一句:
1
|
teacher.setModifyDate( new Date());
|
3.拷贝属性时忽略空值
使用BeanUtils.copyProperties有一个问题就是当src对象的键值为Null时
就会把target对象的对应键值覆盖成空了,这明显不是我们想要的,以下这个方法可以解决
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
/**
* 复制属性,过滤掉不复制的属性
*/
public static void copyBeanProperties(
final Object source, //1,待复制的原始对象
final Object target, //2,复制后的结果对象
//3,获取保存你不需要复制的属性名
final Collection<String> excludes = new ArrayList<String>();
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()]));
}
|
使用案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public static String[] getNullPropertyNames (Object source) {
final BeanWrapper src = new BeanWrapperImpl(source);
java.beans.PropertyDescriptor[] pds = src.getPropertyDescriptors();
Set<String> emptyNames = new HashSet<String>();
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支持的转换类型如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
* 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);
1
2
3
4
5
6
7
8
|
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
1
2
3
4
5
6
7
|
//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方法),以及通过串行化将其返回到远程的客户机的时间总和。所以要小心使用这种威力!有失必有得,反之亦然。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。
原文链接:https://blog.csdn.net/huluwa10526/article/details/108767427