商城首页欢迎来到中国正版软件门户

您的位置:首页 >Java对象克隆方法详解

Java对象克隆方法详解

  发布于2025-09-24 阅读(0)

扫一扫,手机访问

在Java中使用clone()方法需实现Cloneable接口,否则抛出CloneNotSupportedException;2. 正确重写clone()方法以实现对象复制。

如何在Java中使用对象的clone方法

在Java中使用对象的clone()方法,需要遵循一定的规则才能正确实现对象复制。直接调用clone()方法而不做准备会导致编译错误或运行时异常。以下是关键步骤和注意事项。

实现Cloneable接口

Java中的clone()方法定义在Object类中,但默认行为是抛出CloneNotSupportedException。为了让对象支持克隆,必须让类实现Cloneable接口——这是一个标记接口,不包含任何方法。

注意:如果不实现Cloneable接口而调用super.clone(),会抛出异常。

示例:

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

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

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

重写clone方法

虽然Object提供了clone(),但通常需要在子类中重写它,以便返回正确的类型并处理特殊字段。

  • 将访问修饰符从protected改为public,方便外部调用。
  • 返回具体类型而不是Object(可通过泛型或强制转换)。
  • 捕获或声明CloneNotSupportedException

改进后的clone方法:

@Override
public Person clone() {
    try {
        return (Person) super.clone();
    } catch (CloneNotSupportedException e) {
        throw new AssertionError(); // 不应该发生
    }
}

区分浅拷贝与深拷贝

默认的clone()是浅拷贝:基本类型会被复制,但引用类型的字段只是复制了引用,不是新对象。

例如,如果Person包含一个Address对象:

private Address address;

那么super.clone()后,两个Person实例会共享同一个Address对象。

要实现深拷贝,需手动克隆引用字段:

@Override
public Person clone() {
    try {
        Person cloned = (Person) super.clone();
        cloned.address = this.address.clone(); // 假设Address也支持clone
        return cloned;
    } catch (CloneNotSupportedException e) {
        throw new AssertionError();
    }
}

实际使用方式

确保类已正确设置后,就可以安全调用clone方法:

Person p1 = new Person("Alice", 25);
Person p2 = p1.clone(); // 创建副本
p2.setAge(30); // 修改副本不影响原对象

如果是深拷贝,修改嵌套对象也不会影响原对象。

基本上就这些。只要记得实现Cloneable、重写clone()、注意引用类型拷贝问题,就能在Java中正常使用克隆功能。

本文转载于:互联网 如有侵犯,请联系zhengruancom@outlook.com删除。
免责声明:正软商城发布此文仅为传递信息,不代表正软商城认同其观点或证实其描述。
  • 如何在 pytest 中通过组合多个 fixture 实现参数化测试
正版软件
    如何在 pytest 中通过组合多个 fixture 实现参数化测试
    本文介绍一种简洁、可维护的方案:使用@pytest.mark.parametrize驱动测试逻辑,结合fixture依赖链动态生成测试数据,避免直接传递复杂对象,从而安全复用category、product等多层fixture并为不同场景(如NEW/PUBLISHED文章)指定预期状态码。
    14分钟前 0
  • C#访问SysWOW64方法详解 正版软件
    C#访问SysWOW64方法详解
    64位C#程序中Environment.GetFolderPath(Environment.SpecialFolder.System)返回的是重定向后的SysWOW64路径,因其受文件系统重定向机制影响;要真正访问SysWOW64,应直接拼接Path.Combine(Environment.GetEnvironmentVariable("windir"),"SysWOW64"),而非依赖SpecialFolder枚举。
    29分钟前 0
  • Java多态实现简单工厂模式解耦思路 正版软件
    Java多态实现简单工厂模式解耦思路
    简单工厂将对象创建逻辑集中到工厂类中,使业务代码只依赖接口而不关心具体实现,从而实现解耦;新增支付方式只需修改工厂类,无需改动业务代码。
    44分钟前 0
  • 如何为 Go 中的 sql.Null* 类型自定义 JSON 序列化行为
正版软件
    如何为 Go 中的 sql.Null* 类型自定义 JSON 序列化行为
    本文详解如何通过实现encoding.TextMarshaler接口,让sql.NullFloat64等类型在JSON编码时仅输出原始值(如141.0)或null,而非默认的{"Float64":...,"Valid":...}结构。
    59分钟前 0
  • 宝塔自动清理过期备份文件方法 正版软件
    宝塔自动清理过期备份文件方法
    可通过脚本自动清理宝塔面板过期备份:一、用find命令+crontab按修改时间删除;二、用宝塔计划任务执行Shell脚本;三、用Python按文件名日期精准清理;四、调用宝塔API接口远程删除。
    1小时前 15:15 0