一文彻底搞懂深拷贝和浅拷贝的区别

1. 主要区别

深拷贝(Deep Copy)和浅拷贝(Shallow Copy)是在复制对象或数据结构时所使用的两种不同的拷贝方法,它们之间的区别主要在于拷贝的程度和对原始对象内部结构的影响。

1.1 浅拷贝(Shallow Copy)

  • 浅拷贝创建一个新的对象,但是只复制原始对象的基本数据类型的字段或引用(地址),而不复制引用指向的对象。这意味着新对象和原始对象中的引用指向相同的对象。
  • 如果原始对象中的字段是基本数据类型,那么这些字段会被复制到新对象中,而如果字段是引用类型,则新对象和原始对- 象的对应字段将引用同一个对象。
  • 因此,对新对象所做的修改可能会影响到原始对象,因为它们共享相同的引用。

1.2 深拷贝(Deep Copy)

  • 深拷贝创建一个新的对象,并且递归地复制原始对象的所有字段和引用指向的对象,而不仅仅是复制引用本身。
  • 深拷贝会递归复制整个对象结构,包括对象内部的对象,确保新对象和原始对象之间的所有关系都是独立的。
  • 这意味着对新对象所做的修改不会影响到原始对象,因为它们拥有彼此独立的副本。

在这里插入图片描述

2. 浅拷贝示例代码

浅拷贝的示例代码如下,我们这里实现了 Cloneable 接口,并重写了 clone() 方法。clone() 方法的实现很简单,直接调用的是父类 Object 的 clone() 方法。

class Address {
    private String city;

    public Address(String city) {
        this.city = city;
    }

    public String getCity() {
        return city;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}

class Person implements Cloneable {
    private Address address;

    public Person(Address address) {
        this.address = address;
    }

    public Address getAddress() {
        return address;
    }

    @Override
    protected Person clone() throws CloneNotSupportedException {
        return (Person) super.clone();
    }
}

在上面的代码中,Address 类实现了 Cloneable 接口,并重写了 clone() 方法,而 Person 类也实现了 Cloneable 接口并重写了 clone() 方法。

在 Address 类的 clone() 方法中,直接调用了父类 Object 的 clone() 方法。

在 Person 类的 clone() 方法中,也直接调用了父类 Object 的 clone() 方法。

接下来我们进行测试,测试代码如下:

public class Main {
    public static void main(String[] args) {
        try {
            Person person1 = new Person(new Address("武汉"));
            Person person1Copy = person1.clone();
            // true
            System.out.println(person1.getAddress() == person1Copy.getAddress());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

在这个测试中,我们期望输出 true,因为我们实现的是浅拷贝,所以新的 Person 对象和原始的 Person 对象引用的是相同的地址对象。

3. 深拷贝示例代码

这里我们简单对 Person 类的 clone() 方法进行修改,连带着要把 Person 对象内部的 Address 对象一起复制。

class Person implements Cloneable {
    private Address address;

    public Person(Address address) {
        this.address = address;
    }

    public Address getAddress() {
        return address;
    }

    @Override
    protected Person clone() throws CloneNotSupportedException {
        Person person = (Person) super.clone();
        person.address = (Address) this.address.clone();
        return person;
    }
}

在这个修改后的 clone() 方法中,我们首先调用了父类的 clone() 方法创建了一个新的 Person 对象,然后将当前对象的 address 成员变量也进行了克隆,并将克隆后的地址对象赋给新创建的 Person 对象。

接下来,我们来测试一下深拷贝的效果:

public class Main {
    public static void main(String[] args) {
        try {
            Person person1 = new Person(new Address("武汉"));
            Person person1Copy = person1.clone();
            // false
            System.out.println(person1.getAddress() == person1Copy.getAddress());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

在这个测试中,我们期望输出 false,因为我们实现了深拷贝,所以新的 Person 对象和原始的 Person 对象引用的是不同的地址对象。

Python面向对象编程(Object-Oriented Programming,简称OOP)是一种编程范式,它将数据操作封装在对象中,通过对象之间的交互实现程序的设计开发。下面是一些关键概念,帮助你更好地理解Python面向对象编程。 1. 类(Class):类是对象的蓝图或模板,描述了对象的属性行为。它定义了对象的特征方法。例如,我们可以定义一个名为"Car"的类来表示汽车,其中包含属性(如颜色、型号)方法(如加速、刹车)。 2. 对象(Object):对象是类的实例,是具体的实体。通过实例化类,我们可以创建一个对象。例如,我们可以创建一个名为"my_car"的对象,它是基于"Car"类的实例。 3. 属性(Attribute):属性是对象的特征,用于描述对象的状态。每个对象都可以具有一组属性。例如,"Car"类的属性可以包括颜色、型号等。 4. 方法(Method):方法是对象的行为,用于定义对象的操作。每个对象都可以具有一组方法。例如,"Car"类的方法可以包括加速、刹车等。 5. 继承(Inheritance):继承是一种机制,允许我们创建一个新类(称为子类),从现有类(称为父类)继承属性方法。子类可以扩展或修改父类的功能。继承可以实现代码重用层次化设计。 6. 多态(Polymorphism):多态是一种特性,允许不同类的对象对同一方法做出不同的响应。多态提高了代码的灵活性可扩展性。 7. 封装(Encapsulation):封装是一种将数据操作封装在对象中的机制,隐藏了对象的内部实现细节,只暴露必要的接口给外部使用。这样可以保护数据的安全性,提供了更好的模块化代码复用性。 通过理解这些概念,你可以更好地掌握Python面向对象编程。在实践中,你可以使用类来创建对象,操作对象的属性调用对象的方法,通过继承多态实现代码的灵活性可扩展性,通过封装保护数据的安全性提高代码的可维护性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值