2019独角兽企业重金招聘Python工程师标准>>>

对象拷贝类PropertyUtils,BeanUtils,BeanCopier的技术沉淀 博客分类: java

对象拷贝的应用现状简介:

业务系统中经常需要两个对象进行属性的拷贝,不能否认逐个的对象拷贝是最快速最安全的做法,但是当数据对象的属性字段数量超过程序员的容忍的程度,代码因此变得臃肿不堪,使用一些方便的对象拷贝工具类将是很好的选择。

目前流行的较为公用认可的工具类:

Apache的两个版本:(反射机制)

org.apache.commons.beanutils.PropertyUtils.copyProperties(Object dest, Object orig)

org.apache.commons.beanutils.BeanUtils.copyProperties(Object dest, Object orig)

Spring版本:(反射机制)

org.springframework.beans.BeanUtils.copyProperties(Object source, Object target, Class editable, String[] ignoreProperties)

cglib版本:(使用动态代理,效率高)

net.sf.cglib.beans.BeanCopier.copy(Object paramObject1, Object paramObject2, Converter paramConverter)

原理简介

反射类型:(apache)

都使用静态类调用,最终转化虚拟机中两个单例的工具对象。

public BeanUtilsBean()

{

this(new ConvertUtilsBean(), new PropertyUtilsBean());

}

ConvertUtilsBean可以通过ConvertUtils全局自定义注册。

ConvertUtils.register(new DateConvert(), java.util.Date.class);

PropertyUtilsBean的copyProperties方法实现了拷贝的算法。

1、  动态bean:orig instanceof DynaBean:Object value = ((DynaBean)orig).get(name);然后把value复制到动态bean类

2、  Map类型:orig instanceof Map:key值逐个拷贝

3、  其他普通类::从beanInfo【每一个对象都有一个缓存的bean信息,包含属性字段等】取出name,然后把sourceClass和targetClass逐个拷贝

Cglib类型:BeanCopier

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)

1、  获取sourceClass的所有public get 方法-》PropertyDescriptor[] getters

2、  获取TargetClass 的所有 public set 方法-》PropertyDescriptor[] setters

3、  遍历setters的每一个属性,执行4和5

4、  按setters的name生成sourceClass的所有setter方法-》PropertyDescriptor getter【不符合javabean规范的类将会可能出现空指针异常】

5、  PropertyDescriptor[] setters-》PropertyDescriptor setter

6、  将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方法)

备注1

对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));

}

}

备注2

相同属性名,且类型不匹配时候的处理

原因:这两个工具类不支持同名异类型的匹配 !!!【包装类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));

}

}

备注3

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.classfalse); //此处抛出异常创建

}

}

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则忽略掉该字段的操作,这样就可以避免异常的发生。

http://blog.csdn.net/jianhua0902/article/details/8155368

转载于:https://my.oschina.net/xiaominmin/blog/1596987

对象拷贝类PropertyUtils,BeanUtils,BeanCopier的技术沉淀相关推荐

  1. java beanutils 性能_对比BeanUtils、PropertyUtils、BeanCopier的性能消耗

    主要代码 定义2个bean对象: public class copyPropertiesData1 { private Integer id; public Integer getId() { ret ...

  2. 1亿人点赞的晚会,如何做技术沉淀?

    阿里妹导读:今年是双11的第11年,猫晚的第5年.今年的天猫双11狂欢夜(简称"猫晚")有超200个国家和地区通过优酷APP观看猫晚直播.5144万人通过猫晚公益直播间观看明星卖农 ...

  3. 蚂蚁链开源跨链技术,与开发者共享多年区块链技术沉淀

    作为下一代科技革命中的关键技术,区块链这一兵家必争之地,在国内科技企业中又有了新的开放尝试. 4月26日,在"创·见未来"2023蚂蚁集团数字科技开发者大会上,蚂蚁集团旗下蚂蚁链宣 ...

  4. QQ多年IM技术沉淀赋能APP私域社交新体验

    ​APP私域社交新生态 公域流量越来越贵的背景下,金融机构开始主动拥抱私域, 充分挖掘APP私域潜力,深挖服务场景,加强客户连接, 深耕客户体验,强化社交联系. QQ多年IM技术沉淀赋能APP 私域社 ...

  5. Bean复制的几种框架性能比较(BeanUtils、PropertyUtils、BeanCopier)

    作者:费晓晖 cnblogs.com/kaka/archive/2013/03/06/2945514.html 作为一个新员工,一个首要的工作就是阅读别人的代码,阅读代码的诸多好处就不说了,我就直奔主 ...

  6. 对象拷贝之Apache BeanUtils、Spring的BeanUtils、Mapstruct、BeanCopier、PropertieyUtils对比(深拷贝)

    大多时候时候使用的是Apache或Spring``BeanUtils,今天,我们来看一下一个更高效的属性拷贝方式:BeanCopier. https://github.com/cglib/cglib ...

  7. 《UnityAPI.ScriptableObject脚本化对象》(Yanlz+Unity+SteamVR+云技术+5G+AI+VR云游戏+CreateInstance+List+立钻哥哥++OK++)

    <UnityAPI.ScriptableObject脚本化对象> 版本 作者 参与者 完成日期 备注 UnityAPI_ScriptableObject_V01_1.0 严立钻 2020. ...

  8. 玩转娱乐化时代|淘系互动团队几年的技术沉淀+经验都在这!

    作者|渚薰 出品|阿里巴巴新零售淘系技术部 作者简介:淘系技术部-互动-渚薰.12年加入阿里,曾先后参与和担任手淘无线基础架构工作.WEEX 前端框架工作.16年来到淘系互动团队,带领一群志同道合的兄 ...

  9. 爱码物联|9载技术沉淀,只为颠覆而生!

    爱码物联,湖南艾盟科技旗下高端产品系.爱码物联集防伪溯源.数字营销.客户客服.内部管理等四大功能于一体,涵盖全行业.全品类的SaaS服务供应商.爱码物联旨在帮助企业实现防窜物流.万能赋码.精准大数据. ...

最新文章

  1. emmc boot1 boot2 partition
  2. Linux中Shell的命令替换用法笔记
  3. python数独游戏源代码100行_python实现自动解数独小程序
  4. threejs基础示例
  5. 亚马逊、Facebook、Google 等科技公司为何越来越难招到人了?
  6. localStorage、cookie
  7. 安装CentOS的注意事项
  8. [BZOJ1083][SCOI2005]繁忙的都市 最小生成树
  9. Product of Array Except Self
  10. 干掉if-else,试试状态模式!
  11. 免费书籍库,供大家收藏
  12. 登录客户端显示不能连接服务器,客户端不能连接服务器,无法登陆
  13. Java项目:SSM网上超市购物商城管理系统
  14. GVS与唯康教育达成战略合作,共建智能家居人才培养高地
  15. 进销存仓库管理软件的应用场景与效益
  16. [MICO开发帖]MICOkit-Nucleo2.0开发连载贴——1、Micoder开发环境汉化
  17. OpenCV笔记-图像预处理1
  18. FPGA学习之路(一)之D类触发器(Data Flip-Flop/Delay Flip-Flop)搭建
  19. JavaScript历史上的今天是星期几
  20. 数据异常检测方法以及实际应用

热门文章

  1. SAP HUM嵌套HU场景里是否可以直接对内层HU执行VLMOVE?
  2. SAP PM 初级系列7 - 定义工厂区域
  3. 语言模型GPT跨界CV,OpenAI揭示强算力Transformer具有通用性
  4. SAP的SD模块中客户的销售视图--扩充客户销售范围
  5. 深度学习中的双下降现象
  6. 机器学习加深了“知识”和“理解”之间的鸿沟
  7. 中科院张士峰:基于深度学习的通用物体检测算法对比探索
  8. 读了这篇文字,做年薪百万的NLP工程师
  9. CIO职能向流程演进
  10. python中的装饰器介绍