beanutils.copyProperties()的使用总结

首先看一下,BeanUtils.copProperties的方法源码:

abstract class BeanUtils中代码整体分四段截图,去除图片之间的空格就是完整的源码:

上面截图主要的方法操作:

         for (PropertyDescriptor targetPd : targetPds) {
        //获取字段的写方法,--set
            Method writeMethod = targetPd.getWriteMethod();
       //如果方法不为空,且没有忽略
            if (writeMethod != null && (ignoreList == null || (!ignoreList.contains(targetPd.getName())))) {
        //获取原对象的字段
                PropertyDescriptor sourcePd = getPropertyDescriptor(source.getClass(), targetPd.getName());
                if (sourcePd != null) {
                //如果原对象的字段是存在的,获取原对象的这个字段的get方法
                    Method readMethod = sourcePd.getReadMethod();
                    if (readMethod != null &&
                //判断源对象的get的返回值类型和目标对象的set参数是否一致
                    ClassUtils.isAssignable(writeMethod.getParameterTypes()[0], readMethod.getReturnType())) {
                        try {
                        //如果类型一致再判断方法是不是public修饰的
                            if (!Modifier.isPublic(readMethod.getDeclaringClass().getModifiers())) {
                                //如果不是就强制反射
                                readMethod.setAccessible(true);
                            }
                        //然后获取值
                            Object value = readMethod.invoke(source);
                            if (!Modifier.isPublic(writeMethod.getDeclaringClass().getModifiers())) {
                                writeMethod.setAccessible(true);
                            }    
                                //赋值
                            writeMethod.invoke(target, value);
                        }
                        catch (Throwable ex) {
                            throw new FatalBeanException(
                                "Could not copy property '" + targetPd.getName() + "' from source to target", ex);
                        }
                    }
                }
            }
        }

1.属性名一一对应相同情况下,source的bean将覆盖掉目标bean对应的属性值。

bean A                                          bean B

结果:

 

当A类中(source中)有属性值为null的时候:

A类中的address的值为null。

结果:

结论:来源bean中值为null的属性在完成copyPoperties操作后,目标bean中对应的参数值也会变成null

 

2.属性名一一对应相同,参数类型不同。

bean A 中 phone 为String类型                           bean B 中 phone 为int类型 

结果:目标bean B中参数值不受影响,方法copy的前提是参数的类型也要对应相同。:

 

3.属性名大小写问题。

A类中的变量名大写                                    B类中的变量名小写 

相当于参数名不同,找不到参数,所以无法copy成功:

 

4.支持数组,集合

数组:

成功将b中数组类型参数的值改变:

在A,B类中各加入Map类型参数。

同样也成功了:


 

加入list类型参数之后:

最后也成功了:

### 如何使用 Mockito 模拟 `BeanUtils.copyProperties` 方法的调用 Mockito 是一种流行的 Java 测试框架,用于创建模拟对象并定义其行为。然而,在某些情况下,像 `BeanUtils.copyProperties` 这样的静态方法无法被直接模拟,因为 Mockito 默认不支持对静态方法的模拟。 为了实现这一目标,可以借助第三方库 **mock-static** 或者通过 PowerMock 来扩展 Mockito 的功能[^2]。以下是具体的解决方案: #### 使用 PowerMock 扩展 Mockito 模拟静态方法 PowerMock 提供了对静态方法的支持,允许我们模拟诸如 `BeanUtils.copyProperties` 的静态方法调用。下面是具体代码示例: ```java import static org.mockito.Mockito.*; import static org.powermock.api.mockito.PowerMockito.*; import org.apache.commons.beanutils.BeanUtils; import org.junit.Test; import org.junit.runner.RunWith; import org.powermock.core.classloader.annotations.PrepareForTest; import org.powermock.modules.junit4.PowerMockRunner; @RunWith(PowerMockRunner.class) @PrepareForTest(BeanUtils.class) // 声明要模拟的类 public class BeanUtilsTest { @Test public void testCopyProperties() throws Exception { // Mock 静态方法 mockStatic(BeanUtils.class); Object source = new Object(); Object target = new Object(); // 定义当调用 copyProperties 时的行为 doNothing().when(BeanUtils.class, "copyProperties", eq(source), eq(target)); // 调用实际的方法 BeanUtils.copyProperties(source, target); // 验证该方法确实被调用了 verifyStatic(BeanUtils.class, times(1)); BeanUtils.copyProperties(source, target); } } ``` 上述代码展示了如何利用 PowerMock 和 Mockito 结合的方式模拟 `BeanUtils.copyProperties` 的调用,并验证它是否按预期执行[^3]。 --- #### 替代方案:封装静态方法为实例方法 如果不想引入额外依赖(如 PowerMock),可以通过重构代码将静态方法转换为可测试的形式。例如,创建一个包装器类来间接调用 `BeanUtils.copyProperties`: ```java // 封装 BeanUtils 的工具类 public class BeanCopier { public void copyProperties(Object source, Object target) throws Exception { BeanUtils.copyProperties(source, target); } } // 测试代码 @Test public void testCopyPropertiesWithWrapper() throws Exception { BeanCopier beanCopier = mock(BeanCopier.class); // 创建 BeanCopier 的模拟对象 Object source = new Object(); Object target = new Object(); // 定义行为 doNothing().when(beanCopier).copyProperties(eq(source), eq(target)); // 调用方法 beanCopier.copyProperties(source, target); // 验证调用次数 verify(beanCopier, times(1)).copyProperties(source, target); } ``` 这种方式无需修改原有逻辑即可完成单元测试需求[^4]。 --- #### 总结 由于 `BeanUtils.copyProperties` 是静态方法,默认情况下无法直接由 Mockito 模拟。推荐两种解决办法: 1. 使用 PowerMock 扩展 Mockito 功能以支持静态方法模拟; 2. 对现有代码稍作调整,将静态方法封装至实例方法中以便于测试。 这两种方式各有优劣,可根据项目实际情况选择适合的策略。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值