在Java中,如何实现对象的拷贝?
创始人
2025-07-12 00:00:35
0

在Java中,对象的拷贝可以分为浅拷贝(shallow copy)和深拷贝(deep copy)。

  1. 「浅拷贝」:
  • 创建一个新对象,然后将原始对象中的非静态字段复制到新对象,如果字段是值类型,那么对该字段执行逐位复制。如果字段是引用类型,则复制引用但不复制引用的对象。因此,原始对象及其副本引用同一个对象。
  • 在Java中,我们可以通过实现Cloneable接口并重写clone()方法来实现浅拷贝。需要注意的是,Object类中的clone()方法是受保护的,所以我们需要在我们的类中将其重写为public。
  • 另外,如果对象的字段也是需要拷贝的复杂对象,那么可能需要在这些类中也实现Cloneable接口并重写clone()方法。
  1. 「深拷贝」:
  • 创建一个新对象,然后将原始对象中的非静态字段复制到新对象。如果字段是值类型,那么对该字段执行逐位复制。如果字段是引用类型,则递归地复制该字段引用的对象,而不是只复制引用。

  • 在Java中,深拷贝通常需要我们自己写代码来实现,因为Java并没有提供直接实现深拷贝的内置方法。

  • 深拷贝的一个常见实现方式是使用序列化。我们可以将对象写入到一个流中,然后再从流中读取出来,这样得到的就是原对象的一个深拷贝。但是这种方法有一些限制,比如被拷贝的对象以及它引用的所有对象都必须是可序列化的。

注意:Cloneable接口和clone()方法的设计在Java社区中常常被认为是有缺陷的,因为它们有很多问题,比如Cloneable接口没有定义任何方法(它是一个标记接口),clone()方法的访问修饰符是protected,而且它使用的是浅拷贝,这可能会导致意外的对象共享问题。因此,在实际编程中,很多开发者更倾向于自己写代码来实现对象的拷贝,而不是使用Cloneable接口和clone()方法。

  1. 通过实现Cloneable接口并重写clone()方法来实现浅拷贝

我们定一个实体类People,实现了Cloneable接口,并且重写了clone()方法当然也是直接调用的父类的clone()方法。

public class People implements Cloneable {

    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

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

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

测试拷贝:

public class Main {

    public static void main(String[] args) {
        People people = new People();
        people.setName("Reathin");
        people.setAge(30);

        System.out.println("原People" + people.toString());

        try {
            People people1 = people.clone();
            System.out.println("拷贝People" + people1);
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }

    }
}

输出结果如下,我们实现了一次浅拷贝。

图片图片

  1. 通过将原始对象中的非静态字段复制到新对象实现深拷贝
People people2 = new People();
people2.setName(people.getName());
people2.setAge(people.getAge());
System.out.println("深拷贝对象1" + people2);
  1. 通过序列化对象流实现深拷贝
// 序列化对象到字节数组
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
objectOutputStream.writeObject(people);
byte[] serializedData = byteArrayOutputStream.toByteArray();

// 从字节数组中反序列化对象以创建深拷贝
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(serializedData);
ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
People people3 = (People) objectInputStream.readObject();
System.out.println("深拷贝对象2" + people3);

最终输出结果如下:

图片图片

完整示例代码:

public class People implements Cloneable, Serializable {

    private String name;
    private int age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

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

    @Override
    public String toString() {
        return "People{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
public class Main {

    public static void main(String[] args) throws IOException, ClassNotFoundException, CloneNotSupportedException {
        People people = new People();
        people.setName("Reathin");
        people.setAge(30);
        System.out.println("原People" + people.toString());

        People people1 = people.clone();
        System.out.println("浅拷贝People" + people1);

        //深拷贝方式一
        People people2 = new People();
        people2.setName(people.getName());
        people2.setAge(people.getAge());
        System.out.println("深拷贝对象1" + people2);

        //深拷贝方式2
        // 序列化对象到字节数组
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
        objectOutputStream.writeObject(people);
        byte[] serializedData = byteArrayOutputStream.toByteArray();

        // 从字节数组中反序列化对象以创建深拷贝
        ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(serializedData);
        ObjectInputStream objectInputStream = new ObjectInputStream(byteArrayInputStream);
        People people3 = (People) objectInputStream.readObject();
        System.out.println("深拷贝对象2" + people3);

    }
}

开发中可以使用第三方库如Apache Commons Lang的SerializationUtils类或Google的Guava库来实现对象的深拷贝。这些库提供了更加灵活和方便的深拷贝实现方式,同时也提供了更多的自定义选项和错误处理机制。


相关内容

热门资讯

PHP新手之PHP入门 PHP是一种易于学习和使用的服务器端脚本语言。只需要很少的编程知识你就能使用PHP建立一个真正交互的...
网络中立的未来 网络中立性是什... 《牛津词典》中对“网络中立”的解释是“电信运营商应秉持的一种原则,即不考虑来源地提供所有内容和应用的...
各种千兆交换机的数据接口类型详... 千兆交换机有很多值得学习的地方,这里我们主要介绍各种千兆交换机的数据接口类型,作为局域网的主要连接设...
什么是大数据安全 什么是大数据... 在《为什么需要大数据安全分析》一文中,我们已经阐述了一个重要观点,即:安全要素信息呈现出大数据的特征...
全面诠释网络负载均衡 负载均衡的出现大大缓解了服务器的压力,更是有效的利用了资源,提高了效率。那么我们现在来说一下网络负载...
粉嫩如何诠释霸道 东芝M805... “霸道粉”是个什么玩意东芝M805拿过来的时候,笔者扑哧笑了,不是笑这款笔记本,而是笑这款产品的颜色...
如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
30分钟搞定iOS自定义相机 最近公司的项目中用到了相机,由于不用系统的相机,UI给的相机切图,必须自定义才可以。就花时间简单研究...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
P2P的自白|我不生产内容,我... 现在一提起P2P,人们就会联想到正在被有关部门“围剿”的互联网理财服务。×租宝事件使得劳...