首页 > 编程语言 >Java浅拷贝大揭秘:如何轻松复制两个不同对象的某些相同属性

Java浅拷贝大揭秘:如何轻松复制两个不同对象的某些相同属性

时间:2024-05-29 22:23:45浏览次数:23  
标签:p1 Java 对象 clone Person 拷贝 序列化 揭秘

哈喽,大家好,我是木头左!

一、引言

在Java编程中,经常会遇到需要复制一个对象的属性到另一个对象的情况。这时,可以使用浅拷贝(Shallow Copy)来实现这个需求。那么,什么是浅拷贝呢?浅拷贝是指创建一个新对象,然后将原对象的非静态字段复制到新对象中。这样,新对象和原对象就会有相同的字段值。本文将详细介绍如何使用Java实现浅拷贝,并给出代码示例。

二、浅拷贝的原理

浅拷贝的实现原理是通过调用对象的clone()方法来实现的。clone()方法是Object类的一个方法,所有Java类都继承自Object类,因此都可以调用clone()方法。当调用一个对象的clone()方法时,会创建一个新的对象,并将原对象的非静态字段复制到新对象中。需要注意的是,如果字段是引用类型,那么只会复制引用,而不会复制引用指向的对象。这就是浅拷贝的特点。

三、实现浅拷贝的方法

1. 使用clone()方法

要使用clone()方法实现浅拷贝,首先需要让原对象实现Cloneable接口,并重写clone()方法。下面是一个简单的示例:

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();
    }

    // getter和setter方法省略
}

public class Main {
    public static void main(String[] args) {
        Person p1 = new Person("张三", 25);
        try {
            Person p2 = (Person) p1.clone();
            System.out.println("p1: " + p1.getName() + ", " + p1.getAge());
            System.out.println("p2: " + p2.getName() + ", " + p2.getAge());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

p1: 张三, 25
p2: 张三, 25

从运行结果可以看出,使用clone()方法实现了浅拷贝。但是,这种方法有一个缺点,就是无法实现深拷贝。因为当字段是引用类型时,clone()方法只会复制引用,而不会复制引用指向的对象。这就导致了浅拷贝后的新对象和原对象共享同一个引用类型的字段。

2. 使用序列化和反序列化实现浅拷贝

序列化是将对象转换为字节流的过程,反序列化是将字节流转换回对象的过程。通过序列化和反序列化可以实现对象的深拷贝。但是,这种方法只适用于实现了Serializable接口的对象。下面是一个简单的示例:

import java.io.*;

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

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

    // getter和setter方法省略
}

public class Main {
    public static void main(String[] args) {
        Person p1 = new Person("张三", 25);
        try {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bos);
            oos.writeObject(p1);
            oos.flush();
            oos.close();
            bos.close();

            ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bis);
            Person p2 = (Person) ois.readObject();
            System.out.println("p1: " + p1.getName() + ", " + p1.getAge());
            System.out.println("p2: " + p2.getName() + ", " + p2.getAge());
            ois.close();
            bis.close();
        } catch (IOException | ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

运行结果:

p1: 张三, 25
p2: 张三, 25

从运行结果可以看出,使用序列化和反序列化实现了浅拷贝。但是,这种方法的缺点是性能较差,因为序列化和反序列化的过程比较耗时。此外,这种方法还需要对象实现Serializable接口,限制了其适用范围。

四、总结

本文详细介绍了如何使用Java实现浅拷贝,并给出了代码示例。介绍了两种实现浅拷贝的方法:使用clone()方法和序列化与反序列化。虽然这两种方法都可以实现浅拷贝,但它们各有优缺点。使用clone()方法实现浅拷贝简单易用,但无法实现深拷贝;而使用序列化与反序列化实现浅拷贝性能较好,但需要对象实现Serializable接口,限制了其适用范围。在实际开发中,需要根据具体需求选择合适的方法来实现浅拷贝。

我是木头左,感谢各位童鞋的点赞、收藏,我们下期更精彩!

标签:p1,Java,对象,clone,Person,拷贝,序列化,揭秘
From: https://www.cnblogs.com/bigleft/p/18221238

相关文章

  • JAVA每日作业day5.29
    依旧是活力满满的一天奥老铁们。今天学习了数组,数组包括了以下方面:1.动态初始化:自己定义数组的长度,系统决定初始值。2.静态初始化:自己决定数组的初始值,系统决定长度。3.数组的的索引:索引从0开始并逐一增加(每次加1),我们要存储数组的数据时,要用索引来存储,话不多说上代码。......
  • java中String、List、数组之间的转换方式
    在Java中,String、List和数组(如String[])之间的转换是常见的操作。下面是如何在它们之间进行转换的示例。1.String转List通常,你不会直接将一个完整的String转换为List,但你可以将包含多个元素的字符串(如由逗号分隔的字符串)分割成多个部分,并将这些部分添加到List中。Str......
  • 5.29_Java程序流程控制
    CSDN   同C语言的流程同1、补充:1、switch使用时的注意事项1、表达式只能是byte、short、int、char,JDK5开始支持枚举,JDK7开始支持String、不支持double、float、longswitch里面是做分支匹配,也就是可以出现很多分支,如果弄得范围很大,不可能出现这么多分支,int的范围都很......
  • Java数据结构与算法(红黑树)
    前言红黑树是一种自平衡二叉搜索树,确保在插入和删除操作后,树的高度保持平衡,从而保证基本操作(插入、删除、查找)的时间复杂度为O(logn)。实现原理红黑树具有以下性质:每个节点要么是红色,要么是黑色。根节点是黑色的。每个叶子节点(NIL节点,通常是空节点)是黑色的。如果一个节点......
  • Java数据结构与算法(散列表)
    前言散列表是根据关键码值(Keyvalue)而直接进行访问的数据结构。也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度。而key的冲突主要通过链表的方式来处理,后期链表过长情况下可以通过红黑树来优化查询效率。实现原理散列函数(HashFunction):散列函数......
  • Java数据结构与算法(B+树)
    前言B+树(B+Tree)是一种平衡树数据结构,广泛用于数据库和文件系统中。它是一种自平衡的树结构,每个节点包含多个键,并且所有键都是排序的。B+树的叶子节点包含指向相邻叶子节点的指针,这使得范围查询非常高效。B+树的优点1.由于B+树在非叶子结点上不包含真正的数据,只当做索引使用......
  • Erroe:JSON parse error: Cannot deserialize value of type `java.lang.Integer` from
    报错:JSONparseerror:Cannotdeserializevalueoftype`java.lang.Integer`fromObjectvalue(token`JsonToken.START_OBJECT`);nestedexceptioniscom.fasterxml.jackson.databind.exc.MismatchedInputException:Cannotdeserializevalueoftype`java.lang.I......
  • Java项目:205Springboot + vue实现的养老院管理系统(含论文)
    作者主页:夜未央5788 简介:Java领域优质创作者、Java项目、学习资料、技术互助文末获取源码项目介绍基于Springboot+vue实现的养老院管理系统系统包含老人、家属、管理员三个角色系统包含登录、注册、主页、老人管理、家属管理、家属意见管理、寝室管理、老人事故信......
  • 《用ChatGPT轻松搞定Java编程难题:从基础到复杂案例的全面解析》
    ChatGPT国内使用体验点击(文件中并非网站跳转而是详细教程):Docshttps://uajqbcov4oa.feishu.cn/docx/GmeGdznJkoc3nzxHECQcojZ9nXg?from=from_copylink随着人工智能技术的快速发展,越来越多的开发者开始使用ChatGPT来辅助解决编程中的问题。ChatGPT不仅可以快速生成代码,还能进行......
  • java基础
    1.类的概念包:一些接口和类集合在一起,方便使用,类似c语言的头文件使用import关键词,将所用的包导入类:【修饰符】class类名{类体}类中包含构造函数 ,对象(变量),方法等在一个程序中,只有一个pubic类,有一个主类中有main接口,是主程序的接口进入类,用来写一整块的功能【修饰符】包......