服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 基于Beanutils.copyProperties()的用法及重写提高效率

基于Beanutils.copyProperties()的用法及重写提高效率

2021-12-13 13:22菜鸟凯一枚 Java教程

这篇文章主要介绍了Beanutils.copyProperties( )的用法及重写提高效率的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

Beanutils.copyProperties()用法及重写提高效率

特别说明本文介绍的是Spring(import org.springframework.beans.BeanUtils)中的BeanUtils.copyProperties(A,B)方法。是将A中的值赋给B。apache(org.apache.commons.beanutils.BeanUtils)中的BeanUtils.copyProperties(A,B)方法是将B中的值赋值给A。

一、简介

BeanUtils提供对Java反射和自省API的包装。其主要目的是利用反射机制对JavaBean的属性进行处理。我们知道,一个JavaBean通常包含了大量的属性,很多情况下,对JavaBean的处理导致大量get/set代码堆积,增加了代码长度和阅读代码的难度。

二、用法

如果你有两个具有很多相同属性的JavaBean,一个很常见的情况就是Struts里的PO对象(持久对象)和对应的ActionForm。例如:一个用户注册页面,有一个User实体类和一个UserActionForm,我们一般会在Action里从ActionForm构造一个PO对象,传统的方式是使用类似下面的语句对属性逐个赋值:

// 获取 ActionForm 表单数据  
UserActionForm uForm = (UserActionForm) form; 
// 构造一个User对象  
User user = new User();  
// 逐一赋值  
user.setUsername(uForm.getUsername);  
user.setPassword(uForm.getPassword);  
user.setAge(uForm.getAge);    
...........  
...........  
// 然后调用JDBC、或操作Hibernate 持久化对象User到数据库  
HibernateDAO.save(user); 

通过这样的方法如果表单数据N多、100、1000(夸张点。哈哈)、、、、那我们不是要写100、、、1000行set、get了。谁都

不愿意这样做。

而我们使用 BeanUtils.copyProperties() 方法以后,代码量大大的减少,而且整体程序看着也简洁明朗,代码如下:

// 获取 ActionForm 表单数据  
UserActionForm uForm = (UserActionForm) form;      
// 构造一个User对象  
User user = new User();    
BeanUtils.copyProperties(uForm,user);    
// 然后调用JDBC、或操作Hibernate 持久化对象User到数据库  
HibernateDAO.save(user); 

注:如果User和UserActionForm 间存在名称不相同的属性,则BeanUtils不对这些属性进行处理,需要手动处理。例如:

User类里面有个createDate 创建时间字段,而UserActionForm里面无此字段。BeanUtils.copyProperties()不会对此字段做任何处理。必须要自己手动处理。

user.setModifyDate(new Date());  

三、重写

ReflectASM,高性能的反射:

什么是ReflectASM ReflectASM是一个很小的java类库,主要是通过asm生产类来实现java反射,执行速度非常快,看了网上很多和反射的对比,觉得ReflectASM比较神奇,很想知道其原理,下面介绍下如何使用及原理;

public static void main(String[] args) {    
      User user = new User();    
      //使用reflectasm生产User访问类    
      MethodAccess access = MethodAccess.get(User.class);    
      //invoke setName方法name值    
      access.invoke(user, "setName", "张三");    
      //invoke getName方法 获得值    
      String name = (String)access.invoke(user, "getName", null);    
      System.out.println(name);    
  }    

原理

上面代码的确实现反射的功能,代码主要的核心是 MethodAccess.get(User.class);

看了下源码,这段代码主要是通过asm生产一个User的处理类 UserMethodAccess(这个类主要是实现了invoke方法)的ByteCode,然后获得该对象,通过上面的invoke操作user类。

private static Map<Class, MethodAccess> methodMap = new HashMap<Class, MethodAccess>();    
  private static Map<String, Integer> methodIndexMap = new HashMap<String, Integer>(); 
  private static Map<Class, List<String>> fieldMap = new HashMap<Class, List<String>>();  
  public static void copyProperties(Object desc, Object orgi) {  
      MethodAccess descMethodAccess = methodMap.get(desc.getClass());  
      if (descMethodAccess == null) {  
          descMethodAccess = cache(desc);  
      }  
      MethodAccess orgiMethodAccess = methodMap.get(orgi.getClass());  
      if (orgiMethodAccess == null) {  
          orgiMethodAccess = cache(orgi);  
      }  

      List<String> fieldList = fieldMap.get(orgi.getClass());  
      for (String field : fieldList) {  
          String getKey = orgi.getClass().getName() + "." + "get" + field;  
          String setkey = desc.getClass().getName() + "." + "set" + field;  
          Integer setIndex = methodIndexMap.get(setkey);  
          if (setIndex != null) {  
              int getIndex = methodIndexMap.get(getKey);  
              // 参数一需要反射的对象  
              // 参数二class.getDeclaredMethods 对应方法的index  
              // 参数对三象集合  
              descMethodAccess.invoke(desc, setIndex.intValue(),  
                      orgiMethodAccess.invoke(orgi, getIndex));  
          }  
      }  
  }  

  // 单例模式  
  private static MethodAccess cache(Object orgi) {  
      synchronized (orgi.getClass()) {  
          MethodAccess methodAccess = MethodAccess.get(orgi.getClass());  
          Field[] fields = orgi.getClass().getDeclaredFields();  
          List<String> fieldList = new ArrayList<String>(fields.length);  
          for (Field field : fields) {  
              if (Modifier.isPrivate(field.getModifiers())  
                      && !Modifier.isStatic(field.getModifiers())) { // 是否是私有的,是否是静态的  
                  // 非公共私有变量  
                  String fieldName = StringUtils.capitalize(field.getName()); // 获取属性名称  
                  int getIndex = methodAccess.getIndex("get" + fieldName); // 获取get方法的下标  
                  int setIndex = methodAccess.getIndex("set" + fieldName); // 获取set方法的下标  
                  methodIndexMap.put(orgi.getClass().getName() + "." + "get"  
                          + fieldName, getIndex); // 将类名get方法名,方法下标注册到map中  
                  methodIndexMap.put(orgi.getClass().getName() + "." + "set"  
                          + fieldName, setIndex); // 将类名set方法名,方法下标注册到map中  
                  fieldList.add(fieldName); // 将属性名称放入集合里  
              }  
          }  
          fieldMap.put(orgi.getClass(), fieldList); // 将类名,属性名称注册到map中  
          methodMap.put(orgi.getClass(), methodAccess);  
          return methodAccess;  
      }  
  } 

执行1000000条效率80几毫秒,效率已经没问题了。

 

BeanUtils.copyProperties 使用注意

首先结论说在前头, BeanUtils.copyProperties 是浅拷贝 。

基于Beanutils.copyProperties()的用法及重写提高效率

为什么今天我还想把这个BeanUtils.copyProperties 的使用拿出来军训。

因为我意识到了大家(部分)对深拷浅拷还是不清晰,不知道具体的影响。

示例演示

第一个类:

基于Beanutils.copyProperties()的用法及重写提高效率

第二个类:

基于Beanutils.copyProperties()的用法及重写提高效率

注意!! 第二个类里面有使用第一个类。

开始进行示例

  public static void main(String[] args) { 
      /**
       * 模拟数据 A  complexObject
       */
      ComplexObject complexObjectA=new ComplexObject();
      complexObjectA.setNickName("张一");
      SimpleObject simpleObject=new SimpleObject();
      simpleObject.setName("李四");
      simpleObject.setAge(12);
      complexObjectA.setSimpleObject(simpleObject);  
      /**
       * 使用BeanUtils.copyProperties 拷贝 模拟数据 A 生成模拟数据 B
       */
      ComplexObject complexObjectB=new ComplexObject();
      BeanUtils.copyProperties(complexObjectA,complexObjectB);

      System.out.println("拷贝后,查看模拟数据A 和 模拟数据B :");
      System.out.println(complexObjectA.getSimpleObject().toString());
      System.out.println(complexObjectB.getSimpleObject().toString());

      System.out.println("比较模拟数据A 和 模拟数据B 里面的引用对象simple 是否引用地址一样: ");
      System.out.println(complexObjectA.getSimpleObject()==complexObjectB.getSimpleObject()); 

      System.out.println("修改拷贝出来的模拟数据B里面的引用对象simple的属性 age 为 888888");
      complexObjectB.getSimpleObject().setAge(888888);

      System.out.println("修改后,观察原数据A 和拷贝出来的数据 B 里面引用的 对象 simple的属性 age:");
      System.out.println(complexObjectA.getSimpleObject().toString());
      System.out.println(complexObjectB.getSimpleObject().toString()); 
  }

基于Beanutils.copyProperties()的用法及重写提高效率

最后强调

如果你是使用BeanUtils.copyProperties 进行对象的拷贝复制, 一定要注意!

  • 第一点 、你所拷贝的对象内包不包含 其他对象的引用。
  • 第二点、如果包含,那么接下来的方法里无论是操作原对象还是操作拷贝出来的对象是否涉及到 对 对象内 的 那个其他对象的 值的修改 。
  • 第三点、如果涉及到, 修改了,会不会影响到其他方法 对 修改值的使用情况。

就如文中例子, 如果传入过来的一个复杂对象数据A 里面引用了一个 user对象年龄age是10;拷贝出一份数据B后, 操作 数据B的方法把 年龄age改成了88888;

那么后续其他方法用到数据A ,想用的是最初始的 age 为10 ,那么就用不到了,因为浅拷贝的原因受影响,age都变成88888 了。

ok,就提到这。以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家~

原文链接:https://blog.csdn.net/qq_37782076/article/details/86605282

延伸 · 阅读

精彩推荐
  • Java教程Java实现抢红包功能

    Java实现抢红包功能

    这篇文章主要为大家详细介绍了Java实现抢红包功能,采用多线程模拟多人同时抢红包,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙...

    littleschemer13532021-05-16
  • Java教程xml与Java对象的转换详解

    xml与Java对象的转换详解

    这篇文章主要介绍了xml与Java对象的转换详解的相关资料,需要的朋友可以参考下...

    Java教程网2942020-09-17
  • Java教程Java BufferWriter写文件写不进去或缺失数据的解决

    Java BufferWriter写文件写不进去或缺失数据的解决

    这篇文章主要介绍了Java BufferWriter写文件写不进去或缺失数据的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望...

    spcoder14552021-10-18
  • Java教程小米推送Java代码

    小米推送Java代码

    今天小编就为大家分享一篇关于小米推送Java代码,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧...

    富贵稳中求8032021-07-12
  • Java教程升级IDEA后Lombok不能使用的解决方法

    升级IDEA后Lombok不能使用的解决方法

    最近看到提示IDEA提示升级,寻思已经有好久没有升过级了。升级完毕重启之后,突然发现好多错误,本文就来介绍一下如何解决,感兴趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java8中Stream使用的一个注意事项

    Java8中Stream使用的一个注意事项

    最近在工作中发现了对于集合操作转换的神器,java8新特性 stream,但在使用中遇到了一个非常重要的注意点,所以这篇文章主要给大家介绍了关于Java8中S...

    阿杜7482021-02-04
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    这篇文章主要介绍了Java使用SAX解析xml的示例,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下...

    大行者10067412021-08-30
  • Java教程20个非常实用的Java程序代码片段

    20个非常实用的Java程序代码片段

    这篇文章主要为大家分享了20个非常实用的Java程序片段,对java开发项目有所帮助,感兴趣的小伙伴们可以参考一下 ...

    lijiao5352020-04-06