美文网首页
java 中 deep copy & shallow copy

java 中 deep copy & shallow copy

作者: MikeShine | 来源:发表于2022-03-08 16:25 被阅读0次

1. 写在前面

今天遇到了这样一个问题,事实上这个问题是之前遇到过的。
java 中列表的赋值的问题。
这个问题核心是 deep copy & shallow copy 的问题


2. 情景再现

public class MikeTest {

    public static void main(String[] args) throws NoSuchMethodException {

        class Person{
            private String name;
            private Integer age;

            Person(String name, Integer age){
                this.name = name;
                this.age = age;
            }
        }
        // TODO: 2022/3/8  测试一下这个 bug
        List<Person> rawList = new ArrayList<>();
        Person person1 = new Person("mike",24);
        Person person2 = new Person("John",28);
        rawList.add(person1);
        rawList.add(person2);

        List<Person> updatedList = rawList;
        for (Person person : updatedList){
            person.age = 90;
        }
        System.out.println(updatedList);
        System.out.println(rawList);

    }
}

上述代码运行之后,问题如下:


bug结果

3. 问题原因

List<Person> updatedList = rawList;

这一句代码,事实上是将指针给过来了,因此两个 List 对应的是同样的内容
update 之后,新的变了,原来的也变了(因为是一个)


4. 怎么解

一开始我以为下面的方式就可以解决(因为大一学c++课的时候,记得有这么讲过)

List<Person> updatedList = new ArrayList(rawList);

但是最后搞完了,发现问题还是没有解决


5. deep copy

后续去查了一下,这里其实是 shallow copy & deep copy 的问题。

shallow copy

这就是在上述 3 部分中所描述的,shallow copy 只会将地址指到相同的地方,不会重新创建对象。

deep copy

deep copy 就是要一份新的引用对象。

实现则需要元素类去实现 Cloneable 接口, @Override 其中的 clone() 方法。
看代码


/**
 * 实现 Cloneable 的类 对象是人
 * 完成深拷贝
 * 事实上,为了完成深拷贝,是需要被拷贝的元素具有这样的能力
 *
 * @author mikeshine
 */
@Data
public class Person implements Cloneable{

    /**
     * 名字
     */
    private String name;

    /**
     * 年龄
     */
    private Integer age;

    Person(String name, Integer age){
        this.age = age;
        this.name = name;
    }

    /**
     * clone 方法的核心目的就是 返回一个开辟新地址空间的 对象
     *
     * @return
     * @throws CloneNotSupportedException
     */
    @Override
    protected Person clone() throws CloneNotSupportedException {
        // 首先给新对象开辟一个地址空间
        Person newPerson = (Person)super.clone();
        // 然后依次将属性 copy
        newPerson.setName(this.name);
        newPerson.setAge(this.age);
        return newPerson;
    }
}

使用如下

  List<Person> rawList = new ArrayList<>();
        Person person1 = new Person("mike",24);
        Person person2 = new Person("John",28);
        rawList.add(person1);
        rawList.add(person2);

        List<Person> updatedList = new ArrayList<>();

        for(Person person : rawList){
            // 这里是重点,相当于用 clone() 方法重新生成一个新对象放在新地址
            updatedList.add(person.clone());
        }

相关文章

  • Java------List的深拷贝与浅拷贝

    Java的浅拷贝(Shallow Copy)、深拷贝(Deep Copy)。 浅拷贝(Shallow Copy) ...

  • java 中 deep copy & shallow copy

    1. 写在前面 今天遇到了这样一个问题,事实上这个问题是之前遇到过的。java 中列表的赋值的问题。这个问题核心是...

  • OC基础

    Difference between shallow copy and deep copy?1> 浅拷贝:指...

  • copy与deepcopy的区别

    python中有一个copy模块,他提供shallow copy与deep copy两种方式,shallow co...

  • 转载:[题]

    Difference between shallow copy and deep copy?1> 浅拷贝:指针(地...

  • shallow copy & deep copy

    浅拷贝:只复制指向对象的指针,而不复制引用对象本身。在另一个地方用同一个指针引用该对象。 深拷贝:复制引用对象本身...

  • Shallow Copy & Deep Copy

    对象的拷贝需要首先实现协议浅拷贝:object A和ob...

  • IOS 常见面试题

    1.Difference between shallow copy and deep copy? 浅复制和深复制的...

  • 集锦2 iOS面试

    1.Difference between shallow copy and deep copy? 浅复制和深复制的...

  • 集锦1 iOS面试

    iOS面试题锦集 1.Difference between shallow copy and deep copy?...

网友评论

      本文标题:java 中 deep copy & shallow copy

      本文链接:https://www.haomeiwen.com/subject/fkvarrtx.html