1. 利用apache commons-beanutils的开源实现
BeanUtils.copyProperties(dst, src)。方法能够将源对象和目标对象中相同名称的属性值复制过去。注意的是参数前面的是目标对象,后面是源对象。使用该方法需要注意:不能将入口方法与源对象、目标对象之一放在同一源文件之内,否者将没有任何效果。
使用该方法,代码简洁,但是底层利用反射机制,还有许多校验,性能不是十分好。
2.利用ReflectASM,自己编写copyProperties方法
ReflectASM,高性能的反射:
什么是ReflectASM ReflectASM是一个很小的java类库,主要是通过asm生产类来实现java反射,执行速度非常快,看了网上很多和反射的对比,觉得ReflectASM比较神奇,很想知道其原理,下面介绍下如何使用及原理;
- public static void main(String[] args) {
- User user = new User();
-
- MethodAccess access = MethodAccess.get(User.class);
-
- access.invoke(user, "setName", "张三");
-
- String name = (String)access.invoke(user, "getName", null);
- System.out.println(name);
- }
原理
上面代码的确实现反射的功能,代码主要的核心是 MethodAccess.get(User.class);
看了下源码,这段代码主要是通过asm生产一个User的处理类 UserMethodAccess(这个类主要是实现了invoke方法)的ByteCode,然后获得该对象,通过上面的invoke操作user类。
ASM反射转换:
- 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);
-
-
-
- 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);
- int setIndex = methodAccess.getIndex("set" + fieldName);
- methodIndexMap.put(orgi.getClass().getName() + "." + "get"
- + fieldName, getIndex);
- methodIndexMap.put(orgi.getClass().getName() + "." + "set"
- + fieldName, setIndex);
- fieldList.add(fieldName);
- }
- }
- fieldMap.put(orgi.getClass(), fieldList);
- methodMap.put(orgi.getClass(), methodAccess);
- return methodAccess;
- }
- }