Java中浅拷贝和深拷贝

在Java中,拷贝对象可以通过浅拷贝(Shallow Copy)和深拷贝(Deep Copy)实现。浅拷贝只复制对象的引用,而不复制对象本身。深拷贝则复制对象及其所有引用的对象,创建一个完全独立的副本。

浅拷贝

浅拷贝是指复制对象的引用,而不是对象本身。对于对象内部的引用类型字段,浅拷贝会共享同一个引用。

浅拷贝示例
java 复制代码
class Address implements Cloneable {
    String city;

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

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

    @Override
    public String toString() {
        return city;
    }
}

class Person implements Cloneable {
    String name;
    int age;
    Address address;

    Person(String name, int age, Address address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

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

    @Override
    public String toString() {
        return "Person{name='" + name + "', age=" + age + ", address=" + address + '}';
    }
}

public class ShallowCopyExample {
    public static void main(String[] args) throws CloneNotSupportedException {
        Address address = new Address("New York");
        Person person1 = new Person("John", 25, address);
        Person person2 = (Person) person1.clone();

        System.out.println("Before modifying address:");
        System.out.println("Person 1: " + person1);
        System.out.println("Person 2: " + person2);

        person2.address.city = "San Francisco";

        System.out.println("After modifying address:");
        System.out.println("Person 1: " + person1);
        System.out.println("Person 2: " + person2);
    }
}

代码说明

  1. Address 类 :包含一个 city 字段,实现 Cloneable 接口,并重写 clone 方法。
  2. Person 类 :包含 nameageaddress 字段,实现 Cloneable 接口,并重写 clone 方法。
  3. ShallowCopyExample 类 :展示了浅拷贝的效果。修改 person2address 后,person1address 也会受到影响,因为它们共享同一个 Address 对象。

深拷贝

深拷贝是指复制对象及其所有引用的对象,创建一个完全独立的副本。对于对象内部的引用类型字段,深拷贝会递归地复制它们。

深拷贝示例
java 复制代码
class Address implements Cloneable {
    String city;

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

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

    @Override
    public String toString() {
        return city;
    }
}

class Person implements Cloneable {
    String name;
    int age;
    Address address;

    Person(String name, int age, Address address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

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

    @Override
    public String toString() {
        return "Person{name='" + name + "', age=" + age + ", address=" + address + '}';
    }
}

public class DeepCopyExample {
    public static void main(String[] args) throws CloneNotSupportedException {
        Address address = new Address("New York");
        Person person1 = new Person("John", 25, address);
        Person person2 = (Person) person1.clone();

        System.out.println("Before modifying address:");
        System.out.println("Person 1: " + person1);
        System.out.println("Person 2: " + person2);

        person2.address.city = "San Francisco";

        System.out.println("After modifying address:");
        System.out.println("Person 1: " + person1);
        System.out.println("Person 2: " + person2);
    }
}

代码说明

  1. Address 类 :与浅拷贝示例中相同,实现 Cloneable 接口并重写 clone 方法。
  2. Person 类 :在 clone 方法中,对 address 字段进行深拷贝(递归调用 addressclone 方法)。
  3. DeepCopyExample 类 :展示了深拷贝的效果。修改 person2address 后,person1address 不会受到影响,因为它们拥有独立的 Address 对象。

通过这些示例代码,可以看到浅拷贝和深拷贝在实现和效果上的差异。浅拷贝会共享对象的引用,而深拷贝则创建独立的对象副本。

相关推荐
ChinaRainbowSea16 分钟前
1. 初始 RabbitMQ 消息队列
java·中间件·rabbitmq·java-rabbitmq
lmryBC4925 分钟前
golang接口-interface
java·前端·golang
ゞ 正在缓冲99%…26 分钟前
leetcode75.颜色分类
java·数据结构·算法·排序
橘猫云计算机设计38 分钟前
基于springboot的考研成绩查询系统(源码+lw+部署文档+讲解),源码可白嫖!
java·spring boot·后端·python·考研·django·毕业设计
时光呢43 分钟前
JAVA常见的 JVM 参数及其典型默认值
java·开发语言·jvm
程序媛学姐1 小时前
SpringKafka错误处理:重试机制与死信队列
java·开发语言·spring·kafka
向阳2561 小时前
SpringBoot+vue前后端分离整合sa-token(无cookie登录态 & 详细的登录流程)
java·vue.js·spring boot·后端·sa-token·springboot·登录流程
XiaoLeisj1 小时前
【MyBatis】深入解析 MyBatis XML 开发:增删改查操作和方法命名规范、@Param 重命名参数、XML 返回自增主键方法
xml·java·数据库·spring boot·sql·intellij-idea·mybatis
风象南1 小时前
SpringBoot实现数据库读写分离的3种方案
java·spring boot·后端
振鹏Dong2 小时前
策略模式——本质是通过Context类来作为中心控制单元,对不同的策略进行调度分配。
java·策略模式