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

2024-02-01 10:28

本文主要是介绍BeanUtil.copyProperties(source,target)拷贝List注意事项,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一:抛出问题

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@NoArgsConstructorpublic 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本质上为同一个对象);

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@NoArgsConstructorpublic 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号订单)])}
}

这篇关于BeanUtil.copyProperties(source,target)拷贝List注意事项的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/666974

相关文章

解决SpringBoot启动报错:Failed to load property source from location 'classpath:/application.yml'

《解决SpringBoot启动报错:Failedtoloadpropertysourcefromlocationclasspath:/application.yml问题》这篇文章主要介绍... 目录在启动SpringBoot项目时报如下错误原因可能是1.yml中语法错误2.yml文件格式是GBK总结在启动S

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

java streamfilter list 过滤的实现

《javastreamfilterlist过滤的实现》JavaStreamAPI中的filter方法是过滤List集合中元素的一个强大工具,可以轻松地根据自定义条件筛选出符合要求的元素,本文就来... 目录1. 创建一个示例List2. 使用Stream的filter方法进行过滤3. 自定义过滤条件1. 定

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.

Python中__new__()方法适应及注意事项详解

《Python中__new__()方法适应及注意事项详解》:本文主要介绍Python中__new__()方法适应及注意事项的相关资料,new()方法是Python中的一个特殊构造方法,用于在创建对... 目录前言基本用法返回值单例模式自定义对象创建注意事项总结前言new() 方法在 python 中是一个

python中列表list切分的实现

《python中列表list切分的实现》列表是Python中最常用的数据结构之一,经常需要对列表进行切分操作,本文主要介绍了python中列表list切分的实现,文中通过示例代码介绍的非常详细,对大家... 目录一、列表切片的基本用法1.1 基本切片操作1.2 切片的负索引1.3 切片的省略二、列表切分的高

java两个List的交集,并集方式

《java两个List的交集,并集方式》文章主要介绍了Java中两个List的交集和并集的处理方法,推荐使用Apache的CollectionUtils工具类,因为它简单且不会改变原有集合,同时,文章... 目录Java两个List的交集,并集方法一方法二方法三总结java两个List的交集,并集方法一

Springboot的自动配置是什么及注意事项

《Springboot的自动配置是什么及注意事项》SpringBoot的自动配置(Auto-configuration)是指框架根据项目的依赖和应用程序的环境自动配置Spring应用上下文中的Bean... 目录核心概念:自动配置的关键特点:自动配置工作原理:示例:需要注意的点1.默认配置可能不适合所有场景

Spring Cloud Hystrix原理与注意事项小结

《SpringCloudHystrix原理与注意事项小结》本文介绍了Hystrix的基本概念、工作原理以及其在实际开发中的应用方式,通过对Hystrix的深入学习,开发者可以在分布式系统中实现精细... 目录一、Spring Cloud Hystrix概述和设计目标(一)Spring Cloud Hystr

Java集合中的List超详细讲解

《Java集合中的List超详细讲解》本文详细介绍了Java集合框架中的List接口,包括其在集合中的位置、继承体系、常用操作和代码示例,以及不同实现类(如ArrayList、LinkedList和V... 目录一,List的继承体系二,List的常用操作及代码示例1,创建List实例2,增加元素3,访问元