功能简介java
业务系统中常常须要两个对象进行属性的拷贝,不可否认逐个的对象拷贝是最快速最安全的作法,可是当数据对象的属性字段数量超过程序员的容忍的程度,代码所以变得臃肿不堪,使用一些方便的对象拷贝工具类将是很好的选择。程序员
Apache的两个版本:(反射机制)算法
org.apache.commons.beanutils.PropertyUtils.copyProperties(Object dest, Object orig)spring
org.apache.commons.beanutils.BeanUtils.copyProperties(Object dest, Object orig)sql
Spring版本:(反射机制) 对应的包为:spring-beans-4.0.6.RELEASE.jarexpress
org.springframework.beans.BeanUtils.copyProperties(Object source, Object target, Class editable, String[] ignoreProperties)apache
cglib版本:(使用动态代理,效率高)缓存
net.sf.cglib.beans.BeanCopier.copy(Object paramObject1, Object paramObject2, Converter paramConverter)安全
都使用静态类调用,最终转化虚拟机中两个单例的工具对象。jvm
public BeanUtilsBean()
{
this(new ConvertUtilsBean(), new PropertyUtilsBean());
}
ConvertUtilsBean能够经过ConvertUtils全局自定义注册。
ConvertUtils.register(new DateConvert(), java.util.Date.class);
PropertyUtilsBean的copyProperties方法实现了拷贝的算法。
一、 动态bean:orig instanceof DynaBean:Object value = ((DynaBean)orig).get(name);而后把value复制到动态bean类
二、 Map类型:orig instanceof Map:key值逐个拷贝
三、 其余普通类::从beanInfo【每个对象都有一个缓存的bean信息,包含属性字段等】取出name,而后把sourceClass和targetClass逐个拷贝
copier = BeanCopier.create(source.getClass(), target.getClass(), false);
copier.copy(source, target, null);
Create对象过程:产生sourceClass-》TargetClass的拷贝代理类,放入jvm中,因此建立的代理类的时候比较耗时。最好保证这个对象的单例模式,能够参照最后一部分的优化方案。
建立过程:源代码见jdk:net.sf.cglib.beans.BeanCopier.Generator.generateClass(ClassVisitor)
一、 获取sourceClass的全部public get 方法-》PropertyDescriptor[] getters
二、 获取TargetClass 的全部 public set 方法-》PropertyDescriptor[] setters
三、 遍历setters的每个属性,执行4和5
四、 按setters的name生成sourceClass的全部setter方法-》PropertyDescriptor getter【不符合javabean规范的类将会可能出现空指针异常】
五、 PropertyDescriptor[] setters-》PropertyDescriptor setter
六、 将setter和getter名字和类型 配对,生成代理类的拷贝方法。
Copy属性过程:调用生成的代理类,代理类的代码和手工操做的代码很相似,效率很是高。
陷阱条件 |
Apache- PropertyUtils |
Apache- BeanUtils |
Spring- BeanUtils |
Cglib- BeanCopier |
是否能够扩展 useConvete功能 |
NO |
Yes |
Yes |
Yes,但比较难用 |
(sourceObject,targetObject)的顺序 |
逆序 |
逆序 |
OK
|
OK |
对sourceObject特殊属性的限制:(Date,BigDecimal等)【见备注1】 |
OK |
NO,异常出错 |
OK |
OK |
相同属性名,且类型不匹配时候的处理 【见备注2】 |
异常,拷贝部分属性,很是危险 |
OK,并能进行初级转换,Long和Integer互转 |
异常,拷贝部分属性 |
OK,可是该属性不拷贝 |
Get和set方法不匹配的处理 【见备注3】 |
OK |
OK |
OK |
建立拷贝的时候报错,没法拷贝任何属性(当且仅当sourceClass的get方法超过set方法) |
对targetObject特殊属性的限制:(Date,BigDecimal等) 缘由:dateTimeConveter的conveter没有对null值的处理 |
public class ErrorBeanUtilObject { //此处省略getter,setter方法 private String name; private java.util.Date date; } public class ErrorBeanUtilsTest { public static void main(String args[]) throws Throwable { ErrorBeanUtilObject from = new ErrorBeanUtilObject(); ErrorBeanUtilObject to = new ErrorBeanUtilObject(); //from.setDate(new java.util.Date()); from.setName("TTTT"); org.apache.commons.beanutils.BeanUtils.copyProperties(to, from);//若是from.setDate去掉,此处出现conveter异常 System.out.println(ToStringBuilder.reflectionToString(from)); System.out.println(ToStringBuilder.reflectionToString(to)); } } |
相同属性名,且类型不匹配时候的处理 缘由:这两个工具类不支持同名异类型的匹配 !!!【包装类Long和原始数据类型long是能够的】 |
public class TargetClass { //此处省略getter,setter方法 private Long num; private String name; } public class TargetClass { //此处省略getter,setter方法 private Long num; private String name; } public class ErrorPropertyUtilsTest { public static void main(String args[]) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException { SourceClass from = new SourceClass(); from.setNum(1); from.setName("name"); TargetClass to = new TargetClass(); org.apache.commons.beanutils.PropertyUtils.copyProperties(to, from); //抛出参数不匹配异常 org.springframework.beans.BeanUtils.copyProperties(from, to); //抛出参数不匹配异常 System.out.println(ToStringBuilder.reflectionToString(from)); System.out.println(ToStringBuilder.reflectionToString(to)); } } |
Get和set方法不匹配的处理 |
public class ErrorBeanCopierTest { /** * 从该用例看出BeanCopier.create的target.class 的每个get方法必须有队形的set方法 * @param args */ public static void main(String args[]) { BeanCopier copier = BeanCopier.create(UnSatifisedBeanCopierObject.class, SourceClass.class,false); copier = BeanCopier.create(SourceClass.class, UnSatifisedBeanCopierObject.class, false); //此处抛出异常建立 } } class UnSatifisedBeanCopierObject { private String name; private Long num; public String getName() { return name; } public void setName(String name) { this.name = name; } public Long getNum() { return num; } // public void setNum(Long num) { // this.num = num; // } } |
加强apache的beanUtils的拷贝属性,注册一些新的类型转换 |
public class BeanUtilsEx extends BeanUtils { public static void copyProperties(Object dest, Object orig) { try { BeanUtils.copyProperties(dest, orig); } catch (IllegalAccessException ex) { ex.printStackTrace(); } catch (InvocationTargetException ex) { ex.printStackTrace(); } } static { ConvertUtils.register(new DateConvert(), java.util.Date.class); ConvertUtils.register(new DateConvert(), java.sql.Date.class); ConvertUtils.register(new BigDecimalConvert(), BigDecimal.class); } } |
将beancopier作成静态类,方便拷贝 |
public class BeanCopierUtils { public static Map<String,BeanCopier> beanCopierMap = new HashMap<String,BeanCopier>();
public static void copyProperties(Object source, Object target){ String beanKey = generateKey(source.getClass(), target.getClass()); BeanCopier copier = null; if(!beanCopierMap.containsKey(beanKey)){ copier = BeanCopier.create(source.getClass(), target.getClass(), false); beanCopierMap.put(beanKey, copier); }else{ copier = beanCopierMap.get(beanKey); } copier.copy(source, target, null); } private static String generateKey(Class<?> class1,Class<?>class2){ return class1.toString() + class2.toString(); } } |
修复beanCopier对set方法强限制的约束 |
改写net.sf.cglib.beans.BeanCopier.Generator.generateClass(ClassVisitor)方法 将133行的 MethodInfo write = ReflectUtils.getMethodInfo(setter.getWriteMethod()); 预先存一个names2放入 /* 109 */ Map names2 = new HashMap(); /* 110 */ for (int i = 0; i < getters.length; ++i) { /* 111 */ names2.put(setters[i].getName(), getters[i]); /* */ } 调用这行代码前判断查询下,若是没有改writeMethod则忽略掉该字段的操做,这样就能够避免异常的发生。 |