Spring的BeanUtils.copyProperties的坑

一、问题背景

在做源对象目标对象拷贝时目标对象中继承父类的属性没有成功复制。

二、 Spring 的 BeanUtils.copyProperties 方法

使用 Spring 的 BeanUtils.copyProperties 方法进行属性拷贝时,只会拷贝源对象中定义的属性,而不会拷贝目标对象中继承自父类的属性。因为 BeanUtils.copyProperties() 方法是基于 Java 反射实现的,它只能访问源对象中的属性,无法访问目标对象中继承自父类的属性。

如果需要将源对象中的属性拷贝到目标对象中,包括目标对象中继承自父类的属性,可以使用其他的 Java 对象映射工具,比如 Hutool的 BeanUtil、Apache Commons BeanUtils 和 Dozer 等。这些工具可以通过配置来决定是否拷贝继承自父类的属性。

三、demo

以下是使用 Hutool 中的 BeanUtil.copyProperties() 方法进行属性拷贝的示例代码:

import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.date.DateUtil;

import java.util.*;

public class Example {
    public static void main(String[] args) {
        // 创建一个子类对象
        Child source = new Child();
        source.setPublicField("public field");
        source.setProtectedField("protected field");
        source.setPrivateField("private field");
        source.setDateField(DateUtil.parse("2023-05-22"));
        source.setStringList(CollUtil.newArrayList("a", "b", "c"));
        source.setStringMap(new HashMap<String, String>() {{
            put("key1", "value1");
            put("key2", "value2");
        }});

        // 创建一个父类对象
        Parent target = new Parent();

        // 将子类对象的属性拷贝到父类对象中
        BeanUtil.copyProperties(source, target);

        // 输出父类对象中的属性
        System.out.println("publicField: " + target.getPublicField());
        System.out.println("protectedField: " + target.getProtectedField());
        System.println("privateField: " + target.getPrivateField());
        System.out.println("dateField: " + target.getDateField());
        System.out.println("stringList: " + target.getStringList());
        System.out.println("stringMap: " + target.getStringMap());
    }
}

// 父类
class Parent {
    private String privateField;
    private Date dateField;
    private List<String> stringList;
    private Map<String, String> stringMap;

    public String getPrivateField() {
        return privateField;
    }

    public void setPrivateField(String privateField) {
        this.privateField = privateField;
    }

    public Date getDateField() {
        return dateField;
    }

    public void setDateField(Date dateField) {
        this.dateField = dateField;
    }

    public List<String> getStringList() {
        return stringList;
    }

    public void setStringList(List<String> stringList) {
        this.stringList = stringList;
    }

    public Map<String, String> getStringMap() {
        return stringMap;
    }

    public void setStringMap(Map<String, String> stringMap) {
        this.stringMap = stringMap;
    }
}

// 子类
class Child extends Parent {
    public String publicField;
    protected String protectedField;

    public String getPublicField() {
        return publicField;
    }

    public void setPublicField(String publicField) {
        this.publicfield = publicField;
    }

    public String getProtectedField() {
        return protectedField;
    }

    public void setProtectedField(String protectedField) {
        this.protectedField = protectedField;
    }

    private String privateField;

    public String getPrivateField() {
        return privateField;
    }

    public void setPrivateField(String privateField) {
        this.privateField = privateField;
    }
}

输出结果为:

publicField: public field
protectedField: protected field
privateField: private field
dateField: Mon May 22 00:00:00 CST 2023
stringList: [a, b, c]
stringMap: {key1=value1, key2=value2}

四、补充(Spring 的 BeanUtils.copyProperties与hutool中BeanUtil.copyProperties的区别)

Spring 的 BeanUtils.copyProperties() 和 Hutool 中的 BeanUtil.copyProperties() 都是用于对象属性复制的工具方法,但它们在实现细节和使用方式上有一些区别:

1. 底层实现不同

Spring 的 BeanUtils.copyProperties() 方法是基于 Java 反射实现的,它可以将源对象中的属性拷贝到目标对象中,并支持类型转换和自定义转换器。

Hutool 中的 BeanUtil.copyProperties() 方法则是基于 ASM 字节码操作实现的,它不仅可以将源对象中的属性拷贝到目标对象中,还支持自定义映射规则、字段过滤和类型转换。

2. 使用方式不同

Spring 的 BeanUtils.copyProperties() 方法的使用方式如下:

BeanUtils.copyProperties(source, target);

其中,source 是源对象,target 是目标对象。

Hutool 中的 BeanUtil.copyProperties() 方法的使用方式如下:

BeanUtil.copyProperties(source, target, ignoreNullValue);

其中,source 是源对象,target 是目标对象,ignoreNullValue 是一个布尔值,表示是否忽略源对象中值为 null 的属性。如果 ignoreNullValue 为 true,则会忽略源对象中值为 null 的属性,不会拷贝到目标对象中;如果 ignoreNullValue 为 false,则会将源对象中值为 null 的属性拷贝到目标对象中。

3. 支持的类型不同

Spring 的 BeanUtils.copyProperties() 方法支持的类型非常广泛,包括 Java 基本类型、字符串、日期、集合、数组等等。

Hutool 中的 BeanUtil.copyProperties() 方法支持的类型也很广泛,包括 Java 基本类型、字符串、日期、集合、数组等等,但是它还支持一些其他类型,比如枚举、Map、JSONObject 等等。

### BeanUtils.copyProperties 常见问题及解决方法 #### 属性访问器缺失引发异常 当目标类和源类的属性未提供 `getter` 和 `setter` 方法时,调用 `BeanUtils.copyProperties()` 将会抛出 `NoSuchMethodException` 异常[^2]。为了防止这种情况发生,建议确保所有需要被复制的字段都已定义相应的存取方法。 #### 性能考量 鉴于 `BeanUtils.copyProperties()` 利用了反射技术来进行操作,在面对大规模的数据迁移任务时可能构成性能障碍[^4]。针对高频率或大数据量的应用环境,推荐评估并选用更为高效的替代方案,比如手写赋值逻辑或是借助第三方库(例如 MapStruct 或 Dozer),它们通常能够带来更好的执行效率。 #### 类型不匹配处理 有时两个实体间存在同名但不同类型的成员变量;此时即使提供了完整的 getter/setter 接口也可能无法顺利完成映射过程。遇到此类情况应提前校验待转换对象结构的一致性,并通过自定义编辑器或者类型处理器来适配差异化的数据格式。 ```java // 自定义 PropertyEditor 支持特定类型转换 public class CustomDateEditor extends PropertyEditorSupport { private final SimpleDateFormat format; public CustomDateEditor(String pattern) { this.format = new SimpleDateFormat(pattern); } @Override public void setAsText(String text) throws IllegalArgumentException { try { setValue(format.parse(text)); } catch (ParseException e) { throw new IllegalArgumentException(e.getMessage(), e); } } } ``` #### 集合与复杂嵌套对象的支持 对于包含集合或其他复合类型作为其组成部分的目标/源头实例来说,默认行为下的 `copyProperties` 只做浅层拷贝而不深入内部节点进行深克隆。这可能导致意外的结果特别是涉及到可变状态共享的情形下。因此有必要根据实际需求调整策略以实现深层次的对象图遍历与重建工作。 ```java // 手动处理复杂的嵌套对象 public static <T> T deepCopy(Object source, Class<T> targetClass){ if(source == null || !source.getClass().isAnnotationPresent(ComplexObject.class)){ return org.springframework.beans.BeanUtils.copyProperties(source,targetClass.newInstance()); }else{ // 实现深度复制逻辑... } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值