当前位置: 首页 > 工具软件 > ReflectASM > 使用案例 >

使用ReflectASM替代BeanUtils实现对象拷贝

皇甫卓君
2023-12-01

我们将一个类的值要复制到另一个类时,除了繁琐的get/set方法之外,经常会使用到BeanUtils的copyProperties方法,这样会变得非常方便。
注意:避免用 Apache Beanutils 进行属性的 copy,Apache BeanUtils 性能较差
但我在使用的过程中也遇到了一些特殊的需要,比如A类的id属性没有值,而B类的id属性有值,结果使用BeanUtils的copyProperties方法将A拷贝到B之后,B的id值就被覆盖掉了,所以想到了在copyProperties方法基础之上改造一下,由于ReflectASM反射工具的性能很高,所以基于ReflectASM实现了一个对象拷贝工具。

<dependency>
    <groupId>com.esotericsoftware</groupId>
    <artifactId>reflectasm</artifactId>
    <version>1.11.9</version>
</dependency>
import com.esotericsoftware.reflectasm.MethodAccess;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.BeansException;
import org.springframework.beans.FatalBeanException;
import org.springframework.lang.Nullable;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import org.springframework.util.StringUtils;

import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.Arrays;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

/**
 * 基于ReflectASM的对象拷贝工具
 * Created by yangxiangjun on 2020/12/18.
 */
public class CopyUtil {
    private static ConcurrentMap<Class, MethodAccess> localCache = new ConcurrentHashMap<>();

    public static MethodAccess get(Class clazz) {
        if (localCache.containsKey(clazz)) {
            return localCache.get(clazz);
        }

        MethodAccess methodAccess = MethodAccess.get(clazz);
        localCache.putIfAbsent(clazz, methodAccess);
        return methodAccess;
    }

    public static void copyProperties(Object source, Object target) {
        copyProperties(source, target, false, (String[]) null);
    }

    public static void copyProperties(Object source, Object target, String... ignoreProperties) {
        copyProperties(source, target, false, ignoreProperties);
    }

    /**
     * 使用ReflectASM实现的拷贝方法
     * @param source
     * @param target
     * @param ignoreNull
     * @param ignoreProperties
     */
    public static void copyProperties(Object source, Object target, boolean ignoreNull, String... ignoreProperties) {
        Assert.notNull(source, "Source must not be null");
        Assert.notNull(target, "Target must not be null");

        MethodAccess sourceMethodAccess = get(source.getClass());
        MethodAccess targetMethodAccess = get(target.getClass());

        Field[] declaredFields = source.getClass().getDeclaredFields();
        List<String> ignoreList = ignoreProperties != null ? Arrays.asList(ignoreProperties) : null;
        for (Field field : declaredFields) {
            if (targetMethodAccess != null && (ignoreList == null || !ignoreList.contains(field.getName()))) {
                //得到获取属性值得方法
                String getKey;
                if (field.getType() == boolean.class) {
                    getKey = "is" + StringUtils.capitalize(field.getName());
                } else {
                    getKey = "get" + StringUtils.capitalize(field.getName());
                }

                //忽略引用数据类型的空值
                Object value = sourceMethodAccess.invoke(source, getKey, null);
                if (ignoreNull && value == null) {
                    continue;
                }

                //将源目标的值写到目标类中
                String setKey = "set" + StringUtils.capitalize(field.getName());
                int index = targetMethodAccess.getIndex(setKey);
                targetMethodAccess.invoke(target, index, value);
            }
        }
    }

    /**
     * 改造BeanUtils的copyProperties方法
     * @param source
     * @param target
     * @param editable
     * @param ignoreNull
     * @param ignoreProperties
     * @throws BeansException
     */
    public static void copyProperties(Object source, Object target, @Nullable Class<?> editable, boolean ignoreNull, @Nullable String... ignoreProperties) throws BeansException {
        Assert.notNull(source, "Source must not be null");
        Assert.notNull(target, "Target must not be null");
        Class<?> actualEditable = target.getClass();
        if (editable != null) {
            if (!editable.isInstance(target)) {
                throw new IllegalArgumentException("Target class [" + target.getClass().getName() + "] not assignable to Editable class [" + editable.getName() + "]");
            }

            actualEditable = editable;
        }

        PropertyDescriptor[] targetPds = BeanUtils.getPropertyDescriptors(actualEditable);
        List<String> ignoreList = ignoreProperties != null ? Arrays.asList(ignoreProperties) : null;
        PropertyDescriptor[] var7 = targetPds;
        int var8 = targetPds.length;

        for(int var9 = 0; var9 < var8; ++var9) {
            PropertyDescriptor targetPd = var7[var9];
            Method writeMethod = targetPd.getWriteMethod();
            if (writeMethod != null && (ignoreList == null || !ignoreList.contains(targetPd.getName()))) {
                PropertyDescriptor sourcePd = BeanUtils.getPropertyDescriptor(source.getClass(), targetPd.getName());
                if (sourcePd != null) {
                    Method readMethod = sourcePd.getReadMethod();
                    if (readMethod != null && ClassUtils.isAssignable(writeMethod.getParameterTypes()[0], readMethod.getReturnType())) {
                        try {
                            if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers())) {
                                readMethod.setAccessible(true);
                            }

                            Object value = readMethod.invoke(source);

                            if (ignoreNull && value == null){
                                continue;
                            }

                            if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers())) {
                                writeMethod.setAccessible(true);
                            }

                            writeMethod.invoke(target, value);
                        } catch (Throwable var15) {
                            throw new FatalBeanException("Could not copy property '" + targetPd.getName() + "' from source to target", var15);
                        }
                    }
                }
            }
        }

    }
}

经过对比,我发现我实现的拷贝方法在数据量非常庞大的情况下,比如执行100W次,性能其实要比BeanUtils的要低很多,这与ReflectASM的高性能这个事实不相符,若有大佬清楚其中缘由,请评论区帮小编释疑

 类似资料: