java ArrayList的深拷贝与浅拷贝问题

 更新时间:2023年05月31日 10:12:58   作者:kiraraLou  
这篇文章主要介绍了java ArrayList的深拷贝与浅拷贝问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

一、前言

ArrayList是我们经常会用到的集合类,有时候我们为了要不改变原来的数据需要重新拷贝一个新的ArrayList,今天在使用ArrayList拷贝时遇到了一些问题,这里整理并记录一下。

二、准备

首先: ArrayList的常见的拷贝方法有很多,其中都是浅拷贝

这里介绍几种浅拷贝的方式:

1.通过构造函数方法拷贝:

List<Integer> newList = new ArrayList<>(list);

2.addAll()方法

List<Integer> newList = new ArrayList<>();
newList.addAll(list);

3.Collections.copy方法

List<Integer> newList = new ArrayList<>();
newList.addAll(list);
Collections.copy(newList, list)

4.stream 方法

java 8 的新特性

List<Integer> newList = list.stream().collect(toList());

另外一点

clone()方式有些特殊,最开始我以为通过clone()是实现深拷贝。

但其实clone()也是浅拷贝,原因如下:

因为通常我们使用的类型是Interger或者String类型的List,Interger和String类型都是不可变类,那么只需要通过浅拷贝拷贝一层即可。

给人的感觉是完全重新生成了一个新的ArrayList。

但是如果我们将类型改成我们自己的类型时,就会出问题。

三、测试

将类型改成对象,在试一下:

// 模拟些数据
Shard shard1 = new Shard(1,"张三","node1");
Shard shard2 = new Shard(2,"李四","node2");
Shard shard3 = new Shard(3,"王五","node3");
List<Shard> list = Arrays.asList(shard1, shard2, shard3);
// 拷贝一个新的list
List<Shard> newList = new ArrayList<>();
newList.addAll(list);
Collections.copy(newList, list);
// 修改新的list里数据
newList.forEach(e -> e.setShardNum(4));
// 遍历旧的list
list.forEach(e -> System.out.println(e.getShardNum()));

结果:

4
4
4

可以看出这样的List拷贝都是浅拷贝,都是拷贝的对象的引用,并没有真正的去深拷贝。

大家可以试试别的方法,应该都是不行的。

四、深拷贝

那么如何实现一个深拷贝,网上的推荐是使用序列化方法可以实现深拷贝。

代码逻辑贴下:

public class CloneUtil {
    @SuppressWarnings("unchecked")
    public static <T extends Serializable> T clone(T obj){
        T cloneObj = null;
        //写入字节流
        try {
            ByteArrayOutputStream out = new ByteArrayOutputStream();
            ObjectOutputStream obs = new ObjectOutputStream(out);
            obs.writeObject(obj);
            obs.close();
            //分配内存,写入原始对象,生成新对象
            ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(ios);
            //返回生成的新对象
            cloneObj = (T) ois.readObject();
            ois.close();
        }catch(IOException e){
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        return cloneObj;
    }
}

使用:

for(Shard shard: list) {
  Shard newShard = CloneUtil.clone(shard);
    newList.add(newShard);
}

注意点:所有需要拷贝到的对象,通通要实现Serializable

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 浅谈三分钟学习Java泛型中T、E、K、V、?的含义

    浅谈三分钟学习Java泛型中T、E、K、V、?的含义

    这篇文章主要介绍了浅谈三分钟学习Java泛型中T、E、K、V、?的含义,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • Spring AOP 切面@Around注解的用法说明

    Spring AOP 切面@Around注解的用法说明

    这篇文章主要介绍了Spring AOP 切面@Around注解的用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-02-02
  • IDEA项目中配置Maven镜像源(下载源)的详细过程

    IDEA项目中配置Maven镜像源(下载源)的详细过程

    Maven是一个能使我们的java程序开发节省时间和精力,是开发变得相对简单,还能使开发规范化的工具,下面这篇文章主要给大家介绍了关于IDEA项目中配置Maven镜像源(下载源)的详细过程,需要的朋友可以参考下
    2024-02-02
  • 如何使用JavaMail发送邮件

    如何使用JavaMail发送邮件

    这篇文章主要教大家如何使用JavaMail发送邮件在web应用中,实现用户注册成功之后,将用户的注册信息以Email的形式发送到用户的注册邮箱当中,感兴趣的小伙伴们可以参考一下
    2015-12-12
  • JAVA中static方法的用法实例详解

    JAVA中static方法的用法实例详解

    这篇文章主要介绍了JAVA中static方法的用法,结合实例形式较为详细的分析了Java中static方法的功能、使用技巧与相关注意事项,需要的朋友可以参考下
    2015-12-12
  • Java打成各种压缩包的方法详细汇总

    Java打成各种压缩包的方法详细汇总

    在工作过程中,需要将一个文件夹生成压缩文件,然后提供给用户下载,下面这篇文章主要给大家介绍了关于Java打成各种压缩包的相关资料,文中通过代码介绍的非常详细,需要的朋友可以参考下
    2024-06-06
  • java 重定义数组的实现方法(与VB的ReDim相像)

    java 重定义数组的实现方法(与VB的ReDim相像)

    java 重定义数组的实现方法(与VB的ReDim相像),需要的朋友可以参考一下
    2013-04-04
  • Java虚拟机内存结构及编码实战分享

    Java虚拟机内存结构及编码实战分享

    这篇文章主要介绍了Java虚拟机内存结构及编码实战分享,文章围绕详细主题展开相关资料具有一定的参考价值,需要的小伙伴可以参考一下
    2022-04-04
  • Java源码解析ThreadLocal及使用场景

    Java源码解析ThreadLocal及使用场景

    今天小编就为大家分享一篇关于Java源码解析ThreadLocal及使用场景,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • Java线程间协作wait、notify和notifyAll详解

    Java线程间协作wait、notify和notifyAll详解

    这篇文章主要介绍了Java线程间协作wait、notify和notifyAll详解,在 Java 中可以用 wait、notify 和 notifyAll 来实现线程间的通信,尽管关于wait和notify的概念很基础,它们也都是Object类的函数,但用它们来写代码却并不简单,,需要的朋友可以参考下
    2023-10-10

最新评论