cloneable接口

Cloneable 接口是 Java 标准库中的一个标记接口,用于指示一个类的对象能够被合法地克隆。克隆是指创建一个对象的副本,即一个新的对象,其内容与原对象相同。Cloneable 接口本身没有方法,它只是一个标记,表示实现这个接口的类支持克隆操作。

Cloneable 接口的用途

Cloneable 接口的主要用途是与 Object 类中的 clone() 方法配合使用:

  1. 表示类支持克隆

    • 如果一个类实现了 Cloneable 接口,那么表示该类的对象可以被克隆。
    • clone() 方法在调用时会检查对象是否实现了 Cloneable 接口。如果没有实现,则会抛出 CloneNotSupportedException 异常。
  2. 支持浅克隆

    • Object 类的 clone() 方法通过直接复制对象的字段来实现浅克隆。
    • 如果对象的字段是基本数据类型或不可变对象(如 String),那么浅克隆已经足够。
    • 如果对象的字段是可变对象(如数组或自定义对象),则需要实现深克隆。

Cloneable 接口的实现原理

实现 Cloneable 接口并使用 clone() 方法通常需要以下步骤:

  1. 实现 Cloneable 接口

    • 类需要实现 Cloneable 接口,表示该类支持克隆操作。
  2. 重写 clone() 方法

    • 在类中重写 clone() 方法,并调用 super.clone() 方法来执行实际的克隆操作。
  3. 处理异常

    • clone() 方法需要处理 CloneNotSupportedException 异常。
  4. 可选:实现深克隆

    • 如果需要实现深克隆,需在 clone() 方法中手动克隆可变对象的字段。

示例代码

以下是一个实现 Cloneable 接口并支持克隆操作的简单示例:

class Person implements Cloneable {
    private String name;
    private int age;

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

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone(); // 调用 Object 类的 clone() 方法
    }

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

    public static void main(String[] args) {
        try {
            Person person1 = new Person("Alice", 30);
            Person person2 = (Person) person1.clone(); // 克隆对象

            System.out.println(person1);
            System.out.println(person2);

            System.out.println("person1 == person2: " + (person1 == person2)); // false
            System.out.println("person1.equals(person2): " + person1.equals(person2)); // true
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

在这个示例中:

  • Person 类实现了 Cloneable 接口,并重写了 clone() 方法。
  • clone() 方法中调用了 super.clone(),这会调用 Object 类的 clone() 方法,创建一个新的 Person 对象,其字段值与原对象相同。
  • 克隆操作后,person1person2 是两个不同的对象,但它们的内容是相同的。

深克隆示例

如果类中包含可变对象,需要实现深克隆,如下所示:

class Address implements Cloneable {
    private String city;

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

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

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

    @Override
    public String toString() {
        return "Address{city='" + city + "'}";
    }
}

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

    public 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 static void main(String[] args) {
        try {
            Address address = new Address("New York");
            Person person1 = new Person("Alice", 30, address);
            Person person2 = (Person) person1.clone();

            System.out.println(person1);
            System.out.println(person2);

            person2.address.setCity("San Francisco");

            System.out.println("After modifying address:");
            System.out.println(person1);
            System.out.println(person2);

        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

在这个深克隆示例中:

  • Address 类也实现了 Cloneable 接口,并重写了 clone() 方法。
  • Person 类的 clone() 方法中,不仅克隆了 Person 对象本身,还克隆了其包含的 Address 对象,确保 Person 对象的深克隆。

通过实现 Cloneable 接口和重写 clone() 方法,Java 对象可以支持浅克隆或深克隆,根据具体需求灵活处理对象的复制。

深克隆(Deep Copy)和浅克隆(Shallow Copy)是两种不同的对象复制方式。它们的主要区别在于复制的深度,即对于引用类型字段(如对象、数组等)的处理方式不同。

深浅克隆

浅克隆

浅克隆是指创建一个新对象,这个新对象的所有字段都与原对象的字段值相同。但对于引用类型字段,新对象只是复制了引用,而不是引用指向的实际对象。换句话说,浅克隆后的新对象和原对象共享同一个引用对象。

示例

假设有一个 Person 类,包含一个引用类型字段 Address

class Address {
    private String city;

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

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }
}

class Person implements Cloneable {
    private String name;
    private Address address;

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

    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone(); // 浅克隆
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

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

测试浅克隆:

public class ShallowCloneTest {
    public static void main(String[] args) {
        try {
            Address address = new Address("New York");
            Person person1 = new Person("Alice", address);
            Person person2 = (Person) person1.clone();

            System.out.println("Before modification:");
            System.out.println(person1);
            System.out.println(person2);

            person2.getAddress().setCity("San Francisco");

            System.out.println("After modification:");
            System.out.println(person1);
            System.out.println(person2);

        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

输出结果:

Before modification:
Person{name='Alice', address=New York}
Person{name='Alice', address=New York}
After modification:
Person{name='Alice', address=San Francisco}
Person{name='Alice', address=San Francisco}

可以看到,修改 person2 的地址会影响 person1,这是因为浅克隆只是复制了引用,两个对象共享同一个 Address 实例。

深克隆

深克隆是指创建一个新对象,这个新对象的所有字段都与原对象的字段值相同。对于引用类型字段,新对象会创建一个新的实例,并复制引用对象的内容。这样,深克隆后的新对象和原对象是完全独立的。

示例

Person 类实现深克隆:

class Person implements Cloneable {
    private String name;
    private Address address;

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

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Person cloned = (Person) super.clone();
        cloned.address = new Address(this.address.getCity()); // 深克隆
        return cloned;
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

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

测试深克隆:

public class DeepCloneTest {
    public static void main(String[] args) {
        try {
            Address address = new Address("New York");
            Person person1 = new Person("Alice", address);
            Person person2 = (Person) person1.clone();

            System.out.println("Before modification:");
            System.out.println(person1);
            System.out.println(person2);

            person2.getAddress().setCity("San Francisco");

            System.out.println("After modification:");
            System.out.println(person1);
            System.out.println(person2);

        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

输出结果:

Before modification:
Person{name='Alice', address=New York}
Person{name='Alice', address=New York}
After modification:
Person{name='Alice', address=New York}
Person{name='Alice', address=San Francisco}

可以看到,修改 person2 的地址不会影响 person1,这是因为深克隆创建了 Address 的新实例,每个 Person 对象有自己的 Address 实例。

总结

  • 浅克隆:复制对象时,只复制原对象的基本类型字段和引用类型字段的引用。克隆对象与原对象共享同一个引用对象。
  • 深克隆:复制对象时,除了复制基本类型字段外,对于引用类型字段,也会创建新的实例并复制其内容。克隆对象与原对象完全独立。
  • 18
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值