java深拷贝/浅拷贝

类实现Cloneable接口,重写clone方法、方法内容默认调用父类的clone方法。

  1. 浅拷贝

    1. 基础类型的变量拷贝之后是独立的,不会随着源变量变动而变
    2. String类型拷贝之后也是独立的(final)
    3. 引用类型拷贝的是引用地址,拷贝前后的变量引用同一个堆中的对象
  2. 深拷贝
    1. 变量的所有引用类型变量(除了String)都需要实现Cloneable(数组可以直接调用clone方法),clone方法中,引用类型需要各
    2. 自调用clone,重新赋值
    static void cloneTest() throws CloneNotSupportedException {
        Student student = new Student(1, "张", 10, new School(1, "清华"));

        System.out.println(student);
        Student clone = student.clone();

        clone.setId(2);
        clone.setName("li");
        clone.setAge(18);
        clone.getSchool().id = 2;
        clone.getSchool().name = "beida";

        System.out.println("clone---->" + clone);
        System.out.println("student-->" + student);
    }    

    @AllArgsConstructor
    @Data
    static class Student implements Cloneable {

        int id;
        String name;
        Integer age;
        School school;

        @Override
        public String toString() {
            return "Student{" +
                    "id=" + id +
                    ", name='" + name + '\'' +
                    ", age=" + age +
                    ", school=" + school +
                    '}';
        }

        @Override
        public Student clone() throws CloneNotSupportedException {
            Student clone = (Student) super.clone(); // 只拷贝基本类型
            // clone.school = clone.getSchool().clone();  // 拷贝引用类型
            return clone;
        }
    }

    @AllArgsConstructor
    static class School implements Cloneable {

        int id;
        String name;
        @Override
        public String toString() {
            return "School{" +
                    "id=" + id +
                    ", name='" + name + '\'' +
                    '}';
        }

        @Override
        public School clone() throws CloneNotSupportedException {
            return (School) super.clone();
        }
    }

如果注释掉这段代码,

 // clone.school = clone.getSchool().clone();  // 拷贝引用类型

输出之后可以发现,基础类型的原值没有改动,但是引用类型的school被改了,这就是浅拷贝。

那么如果取消注释呢?可以看到都没有变化,这就是深拷贝,修改拷贝对象,对原来的对象所有属性都没有影响

java的传参,基本类型和引用类型传参

java在方法传递参数时,是将变量复制一份,然后传入方法体去执行。复制的是栈中的内容

所以基本类型是复制的变量名和值,值变了不影响源变量

引用类型复制的是变量名和值(引用地址),对象变了,会影响源变量(引用地址是一样的)

String:是不可变对象,重新赋值时,会在常量表新生成字符串(如果已有,直接取他的引用地址),将新字符串的引用地址赋值给栈中的新变量,因此源变量不会受影响

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JAVA中的深拷贝浅拷贝是对象拷贝的两种方式。深拷贝是一个整个独立的对象拷贝,会拷贝所有的属性,并拷贝属性指向的动态分配的内存。当对象和它所引用的对象一起拷贝时即发生深拷贝深拷贝相比于浅拷贝速度较慢并且花销较大。要实现深拷贝,可以在拷贝构造方法中,对引用数据类型变量逐一开辟新的内存空间,创建新的对象,这样可以实现深拷贝。而对于一般的拷贝构造,则一定是浅拷贝。此外,可以通过简洁的代码实现深拷贝,但要注意的是,如果某个属性被transient修饰,那么该属性就无法被拷贝了。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [Java深入理解深拷贝浅拷贝区别](https://blog.csdn.net/riemann_/article/details/87217229)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [Java 浅拷贝深拷贝的理解和实现方式](https://blog.csdn.net/weixin_30662011/article/details/96181137)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值