BeanUtil.copyProperties(source,target)拷贝List注意事项

一:抛出问题

java 复制代码
import cn.hutool.core.bean.BeanUtil;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.ArrayList;
import java.util.List;
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
    private String id;
    private String name;
    /* 用于存储用户的订单 */
    private List<Order> orders;

    @Data
    @AllArgsConstructor
    @NoArgsConstructor
  public static class Order{
      /* 订单ID */
      private String orderId;
      /* 订单名称 */
      private String orderName;
   }

    /**
     * BeanUtil.copyProperties(source,target)方法拷贝List类型数据时。对target对象中的List数据修改会影响到source对象中的List数据
     */
    public static void main(String[] args) {
        User source = new User();
             source.setId("001");
             source.setName("001号用户");

        // 构建source对象orders数据
        List<Order> orders = new ArrayList();
        Order order = new Order();
              order.setOrderId("1");
              order.setOrderName("1号订单");
        orders.add(order);
        source.setOrders(orders);
        System.out.println("source = " + source); // 输出:User(id=001, name=001号用户, orders=[User.Order(orderId=1, orderName=1号订单)])

        // 构建拷贝对象
        User target = new User();

        // 属性拷贝
        BeanUtil.copyProperties(source,target);
        System.out.println("target = " + target); // 输出:User(id=001, name=001号用户, orders=[User.Order(orderId=1, orderName=1号订单)])

        // 对target对象中的orders集合数据进行修改,观察source对象中的orders集合数据是否会发送变化
        Order targetOrder = target.getOrders().get(0);
              targetOrder.setOrderId("2");
              targetOrder.setOrderName("2号订单");

         // 输出target对象
        System.out.println("target = " + target); // User(id=001, name=001号用户, orders=[User.Order(orderId=2, orderName=2号订单)])

        /**
         * 输出source对象
         *      发现source对象中的orders数据被修改了。原因在于对于List类型使用BeanUtil.copyProperties(source,target)方法重source拷贝数据到target时。target对象中的orders对象其实是source对象中orders对象的引用(本质上是同一个对象)
         *      所以出现了对target对象中List类型的orders对象数据修改同时也会影响到source对象中的List类型的orders数据。
         */
        System.out.println("source = " + source); // User(id=001, name=001号用户, orders=[User.Order(orderId=2, orderName=2号订单)])
    }
}

二:解决方式

已知上面对target对象中的List数据修改时会影响到source对象中的List数据;也清楚了出现这个问题的原因在于使用copyProperties方法拷贝List属性时只是将source中List的引用拷贝给target中的List(source和target对象中的List本质上为同一个对象);

java 复制代码
import cn.hutool.core.bean.BeanUtil;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.ArrayList;
import java.util.List;
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
    private String id;
    private String name;
    /* 用于存储用户的订单 */
    private List<Order> orders;

    @Data
    @AllArgsConstructor
    @NoArgsConstructor
  public static class Order{
      /* 订单ID */
      private String orderId;
      /* 订单名称 */
      private String orderName;
   }


    public static void main(String[] args) {
        User source = new User();
             source.setId("001");
             source.setName("001号用户");

        // 构建source对象orders数据
        List<Order> orders = new ArrayList();
        Order order = new Order();
              order.setOrderId("1");
              order.setOrderName("1号订单");
        orders.add(order);
        source.setOrders(orders);
        System.out.println("source = " + source); // 输出:User(id=001, name=001号用户, orders=[User.Order(orderId=1, orderName=1号订单)])

        // 构建拷贝对象
        User target = new User();

        // 属性拷贝
        BeanUtil.copyProperties(source,target);

        // 解决方式:使用copyToList拷贝List数据时会创建一个新的List而不是地址值引用(将创建出来新的List对象赋值给target中的List)
        List<Order> newOrders = BeanUtil.copyToList(source.getOrders(), Order.class);
        target.setOrders(newOrders);
        System.out.println("target = " + target); // 输出:User(id=001, name=001号用户, orders=[User.Order(orderId=1, orderName=1号订单)])

        // 对target对象中的orders集合数据进行修改,观察source对象中的orders集合数据是否会发送变化(不会)
        Order targetOrder = target.getOrders().get(0);
              targetOrder.setOrderId("2");
              targetOrder.setOrderName("2号订单");

         // 输出target对象
        System.out.println("target = " + target); // User(id=001, name=001号用户, orders=[User.Order(orderId=2, orderName=2号订单)])

        // 输出source对象
        System.out.println("source = " + source); // User(id=001, name=001号用户, orders=[User.Order(orderId=1, orderName=1号订单)])
    }
}
相关推荐
jzhwolp12 小时前
nginx epoll 里黑科技位运算+指针复用
linux·nginx·架构
NON-JUDGMENTAL13 小时前
在 Ubuntu 上安装 Ollama 并通过 Open WebUI 运行本地大语言模型
linux·ubuntu·语言模型
ZzzZZzzzZZZzzzz…13 小时前
RHCSA---权限管理
linux·运维·权限管理·特殊权限·rhcsa·acl权限·权限掩码原理
TinyPiXOS开发者联盟13 小时前
轻量级嵌入式系统的 Lottie 动画实现
linux·c++·动画·嵌入式开发·lottie·tinypixos·tpgui
海蓝可知天湛13 小时前
Ubuntu24.10禁用该源...+vmware无法复制黏贴“天坑闭环”——从 DNS 诡异解析到 Ubuntu EOL 引发的 apt 404排除折
linux·ubuntu
SoulKuyan14 小时前
android su执行命令
linux·运维·服务器
Zhao_yani14 小时前
Centos 7安装Apache Drill
linux·centos·drill
一叶飘零_sweeeet14 小时前
Linux 安装 Elasticsearch:避坑指南 + 性能调优实战
linux·运维·elasticsearch
梁正雄15 小时前
linux-shell-基础与变量和运算符-1
linux·运维
HIT_Weston15 小时前
23、【Ubuntu】【远程开发】内网穿透:SSH 反向隧道
linux·ubuntu·ssh