在Java中为什么不建议使用Object的clone()来拷贝对象

为什么不建议使用Object的clone()来拷贝对象

前言

最近阅读了《阿里巴巴Java开发手册》一书,书中提到了不推荐使用Object对象的clone()方法来对对象进行拷贝,因为Object的clone()方法默认是浅拷贝,原文如下:

【推荐】慎用Object的clone方法来拷贝对象。
说明:对象的clone方法默认是浅拷贝,若想实现深拷贝需要重写clone方法实现属性对象的拷贝。

浅拷贝与深拷贝

在Java中数据分为基本数据类型引用数据类型,基本数据类型存储在栈中,而引用数据类型则是在栈中存储指向对象的引用地址,数据实际上存储在堆内存中。
在这里插入图片描述
浅拷贝只复制指向某个对象的地址,而不复制对象本身,新旧对象还是共享同一块内存。但深拷贝会另外创造一个一模一样的对象,新对象跟原对象不共享内存,修改新对象不会改到原对象。

clone()方法

java中的clone()方法在Object类中定义,如下图:
在这里插入图片描述
我们可以看到clone()方法使用protected访问修饰符修饰,所以我们可以在Object子类中重写这个方法并使用。

探究

以下实体类使用lombok生成构造器与get、set方法

首先我们创建一个员工类,重写并调用父类Object的clone()

/**
 * @package: com.vinci.testClone
 * @className: Employee
 * @author: Vinci
 * @description: 员工对象
 * @date: 2023/9/22 17:00
 */
@Data
@NoArgsConstructor
@AllArgsConstructor(access = AccessLevel.PUBLIC)
//注意,若使用clone(),要先实现Cloneable接口
public class Employee implements Cloneable{

    private int id;

    private String name;

    private int age;

    private double salary;

    private Company company;

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

}

不过需要注意,若使用clone(),要先实现Cloneable接口,否则会抛出CloneNotSupportedException异常
在这里插入图片描述
创建公司类,作为员工类的引用属性

/**
 * @package: com.vinci.testClone
 * @className: Company
 * @author: Vinci
 * @description:
 * @date: 2023/9/22 17:02
 */
@Data
@NoArgsConstructor
@AllArgsConstructor(access = AccessLevel.PUBLIC)
public class Company {

    private int id;

    private String name;

    private String address;

}

接着我们编写代码测试 clone(),创建一个员工对象,员工对象中包含公司对象

//使用有参构造器初始化对象
Employee employee = new Employee(
        1,"Vinci",23,10000D,
        new Company(
                1,
                "VinciOs",
                "波西米亚"
        )
);
// 输出结果 Employee(id=1, name=Vinci, age=23, salary=10000.0, company=Company(id=1, name=VinciOs, address=波西米亚))
System.out.println(employee);

使用 clone() 方法 拷贝对象

Employee newEmployee = (Employee) employee.clone();
//输出结果 Employee(id=1, name=Vinci, age=23, salary=10000.0, company=Company(id=1, name=VinciOs, address=波西米亚))
System.out.println(newEmployee);

我们可以看到两个对象的输出结果是一致的,那我们比较两个对象的地址呢?

 //比较两个对象的地址
 System.out.println(employee == newEmployee); // false
 //比较两个对象中引用数据类型的地址
 System.out.println(employee.getCompany() == newEmployee.getCompany());//ture

通过比较我们可以发现,两个对象的引用地址不相同,但是对象里包含的引用类型的地址却是相同的。

//如果我们要修改第二个员工的公司呢?
newEmployee.getCompany().setAddress("捷克");

//输出结果 Employee(id=1, name=Vinci, age=23, salary=10000.0, company=Company(id=1, name=VinciOs, address=捷克))
System.out.println(employee);
//输出结果 Employee(id=1, name=Vinci, age=23, salary=10000.0, company=Company(id=1, name=VinciOs, address=捷克))
System.out.println(newEmployee);

结果我们发现 由于这两个对象中的company属性引用的是同一个地址,因此不管修改哪一个另一个都会跟着改变,很容易出现事故
很多时候我们去拷贝对象是希望进行深度拷贝的,因此不建议使用clone()进行拷贝。

实现深度拷贝的两种方式

1、重写clone()方法

修改代码如下:

@Data
@NoArgsConstructor
@AllArgsConstructor(access = AccessLevel.PUBLIC)
//注意,若使用clone(),要先实现Cloneable接口
public class Employee implements Cloneable{

    private int id;

    private String name;

    private int age;

    private double salary;

    private Company company;

    @Override
    protected Object clone() throws CloneNotSupportedException {
//        return super.clone();
        Employee employee = (Employee) super.clone();
        employee.company = (Company) this.company.clone();
        return employee;
    }

}

@Data
@NoArgsConstructor
@AllArgsConstructor(access = AccessLevel.PUBLIC)
public class Company implements Cloneable{

    private int id;

    private String name;

    private String address;

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


}

随后我们再运行同样的代码,发现:
在这里插入图片描述

2、使用序列化实现深度拷贝

需要导入依赖,并将两个类实现Serializable接口

<dependency>
    <groupId>org.apache.commons</groupId>
    <artifactId>commons-lang3</artifactId>
    <version>3.12.0</version>
</dependency>

在这里插入图片描述

 public void serializedCopy(){
      //创建对象
      Employee employee = new Employee(
              1,"Vinci",23,10000D,
              new Company(
                      1,
                      "VinciOs",
                      "波西米亚"
              )
      );
      //序列化对象
      byte[] serialize = SerializationUtils.serialize(employee);
      //反序列化对象
      Employee newEmployee = SerializationUtils.deserialize(serialize);
      System.out.println(newEmployee == employee); // false
}

在这里插入图片描述
测试后我们发现,使用序列化实现的拷贝,完全是全新的对象,是深度拷贝的。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java的克隆(Clone)是一种对象复制的方式,它可以复制一个已有对象的所有属性,然后生成一个全新的对象。这个新对象在内存与原对象完全独立,它们具有相同的属性值,但是修改其一个对象的属性值不会影响另一个对象的属性值。 在Java使用克隆的情况主要有两种: 1. 对象复制:当我们需要在程序创建新的对象,但是这个新对象与已有的对象具有相同的属性值时,我们可以使用克隆来实现。这种情况下,我们需要实现Cloneable接口,并且重写Object类的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 public Object clone() throws CloneNotSupportedException { return super.clone(); } } ``` 2. 防止修改:有时候我们需要传递一个对象给其他方法或者类使用,但是我们不希望这个对象被修改。这个时候,我们可以使用克隆来生成一个新对象,然后传递这个新对象。这种情况下,我们需要在克隆方法返回一个新对象,而不是原对象的引用。 例如: ``` public class Person implements Cloneable { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } @Override public Object clone() throws CloneNotSupportedException { return new Person(this.name, this.age); } } ``` 在这个例子,我们重写了clone()方法,并且返回了一个新的Person对象,这个对象的属性值与原对象相同。当我们需要传递一个Person对象给其他方法或者类使用时,我们可以将这个新对象传递给它们,这样就可以防止这个对象被修改了。 需要注意的是,使用克隆需要注意一些细节,例如如果对象包含引用类型的属性,那么我们需要对这些属性也进行克隆。另外,克隆是一种浅拷贝的方式,如果我们需要深拷贝一个对象,需要使用其他方式。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值