beanutils.copyproperties
一、简介:
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到数据库
- ...
通过这样的方法如果表单数据N多、100、1000(夸张点。哈哈)、、、、那我们不是要写100、、、1000行set、get了。谁都不愿意这样做。
而我们使用 beanutils.copyproperties() 方法以后,代码量大大的减少,而且整体程序看着也简洁明朗,代码如下:
Java代码
- // 获取 ActionForm 表单数据
- UserActionForm uForm = (UserActionForm) form;
- // 构造一个User对象
- User user = new User();
- // 赋值
- BeanUtils.copyProperties(user, uForm);
- // 然后调用JDBC、或操作Hibernate 持久化对象User到数据库
- .......
注:如果User和UserActionForm 间存在名称不相同的属性,则BeanUtils不对这些属性进行处理,需要手动处理。例如:
User类里面有个createDate 创建时间字段,而UserActionForm里面无此字段。BeanUtils.copyProperties()不会对此字段做任何处理。必须要自己手动处理。
用法总结如下:
- BeanUtils.copyProperties("转换后的类", "要转换的类");
[java] view plain copy print?
- propertyutils.copyProperties("转换后的类", "要转换的类");
用法其实很简单,第一个参数是转换后的类,第二个参数是待转换的类
三、重写:
ReflectASM,高性能的反射:
什么是ReflectASM ReflectASM是一个很小的java类库,主要是通过asm生产类来实现java反射,执行速度非常快,看了网上很多和反射的对比,觉得ReflectASM比较神奇,很想知道其原理,下面介绍下如何使用及原理;
[java] view plain copy print?
- 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类。
ASM反射转换:
[java] view plain copy print?
- 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几毫秒,效率已经没问题了;
相关阅读
Spring中提供的属性拷贝的方法BeanUtils.copyProperti
BeanUtils.copyProperties通过java反射将类中当前属性字段对应的内容复制到另外一个类中 //内部都是调用下面的私有方法 1. B