java不同对象之间同名属性值复制

1. 利用apache commons-beanutils的开源实现

BeanUtils.copyProperties(dst, src)。方法能够将源对象和目标对象中相同名称的属性值复制过去。注意的是参数前面的是目标对象,后面是源对象。使用该方法需要注意:不能将入口方法与源对象、目标对象之一放在同一源文件之内,否者将没有任何效果。

使用该方法,代码简洁,但是底层利用反射机制,还有许多校验,性能不是十分好。


2.利用ReflectASM,自己编写copyProperties方法

ReflectASM,高性能的反射:

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

[java]  view plain  copy
 print ?
  1. public static void main(String[] args) {    
  2.         User user = new User();    
  3.         //使用reflectasm生产User访问类    
  4.         MethodAccess access = MethodAccess.get(User.class);    
  5.         //invoke setName方法name值    
  6.         access.invoke(user, "setName""张三");    
  7.         //invoke getName方法 获得值    
  8.         String name = (String)access.invoke(user, "getName"null);    
  9.         System.out.println(name);    
  10.     }    
原理 
   上面代码的确实现反射的功能,代码主要的核心是 MethodAccess.get(User.class); 
看了下源码,这段代码主要是通过asm生产一个User的处理类 UserMethodAccess(这个类主要是实现了invoke方法)的ByteCode,然后获得该对象,通过上面的invoke操作user类。 

ASM反射转换:

[java]  view plain  copy
 print ?
  1. private static Map<Class, MethodAccess> methodMap = new HashMap<Class, MethodAccess>();  
  2.   
  3.     private static Map<String, Integer> methodIndexMap = new HashMap<String, Integer>();  
  4.   
  5.     private static Map<Class, List<String>> fieldMap = new HashMap<Class, List<String>>();  
  6.   
  7.     public static void copyProperties(Object desc, Object orgi) {  
  8.         MethodAccess descMethodAccess = methodMap.get(desc.getClass());  
  9.         if (descMethodAccess == null) {  
  10.             descMethodAccess = cache(desc);  
  11.         }  
  12.         MethodAccess orgiMethodAccess = methodMap.get(orgi.getClass());  
  13.         if (orgiMethodAccess == null) {  
  14.             orgiMethodAccess = cache(orgi);  
  15.         }  
  16.   
  17.         List<String> fieldList = fieldMap.get(orgi.getClass());  
  18.         for (String field : fieldList) {  
  19.             String getKey = orgi.getClass().getName() + "." + "get" + field;  
  20.             String setkey = desc.getClass().getName() + "." + "set" + field;  
  21.             Integer setIndex = methodIndexMap.get(setkey);  
  22.             if (setIndex != null) {  
  23.                 int getIndex = methodIndexMap.get(getKey);  
  24.                 // 参数一需要反射的对象  
  25.                 // 参数二class.getDeclaredMethods 对应方法的index  
  26.                 // 参数对三象集合  
  27.                 descMethodAccess.invoke(desc, setIndex.intValue(),  
  28.                         orgiMethodAccess.invoke(orgi, getIndex));  
  29.             }  
  30.         }  
  31.     }  
  32.   
  33.     // 单例模式  
  34.     private static MethodAccess cache(Object orgi) {  
  35.         synchronized (orgi.getClass()) {  
  36.             MethodAccess methodAccess = MethodAccess.get(orgi.getClass());  
  37.             Field[] fields = orgi.getClass().getDeclaredFields();  
  38.             List<String> fieldList = new ArrayList<String>(fields.length);  
  39.             for (Field field : fields) {  
  40.                 if (Modifier.isPrivate(field.getModifiers())  
  41.                         && !Modifier.isStatic(field.getModifiers())) { // 是否是私有的,是否是静态的  
  42.                     // 非公共私有变量  
  43.                     String fieldName = StringUtils.capitalize(field.getName()); // 获取属性名称  
  44.                     int getIndex = methodAccess.getIndex("get" + fieldName); // 获取get方法的下标  
  45.                     int setIndex = methodAccess.getIndex("set" + fieldName); // 获取set方法的下标  
  46.                     methodIndexMap.put(orgi.getClass().getName() + "." + "get"  
  47.                             + fieldName, getIndex); // 将类名get方法名,方法下标注册到map中  
  48.                     methodIndexMap.put(orgi.getClass().getName() + "." + "set"  
  49.                             + fieldName, setIndex); // 将类名set方法名,方法下标注册到map中  
  50.                     fieldList.add(fieldName); // 将属性名称放入集合里  
  51.                 }  
  52.             }  
  53.             fieldMap.put(orgi.getClass(), fieldList); // 将类名,属性名称注册到map中  
  54.             methodMap.put(orgi.getClass(), methodAccess);  
  55.             return methodAccess;  
  56.         }  
  57.     }  

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值