第35题JAVA高级技术-对象克隆4(序列化与对象克隆)

Posted 小虚竹

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第35题JAVA高级技术-对象克隆4(序列化与对象克隆)相关的知识,希望对你有一定的参考价值。

文章目录

零、前言

​ 今天是学习 JAVA语言 打卡的第35天,每天我会提供一篇文章供群成员阅读( 不需要订阅付钱 ),读完文章之后,按解题思路,自己再实现一遍。在小虚竹JAVA社区 中对应的 【打卡贴】打卡,今天的任务就算完成了。

​ 因为大家都在一起学习同一篇文章,所以有什么问题都可以在群里问,群里的小伙伴可以迅速地帮到你,一个人可以走得很快,一群人可以走得很远,有一起学习交流的战友,是多么幸运的事情。

​ 学完后,自己写篇学习报告的博客,可以发布到小虚竹JAVA社区 ,供学弟学妹们参考。

​ 我的学习策略很简单,题海策略+ 费曼学习法。如果能把这100题都认认真真自己实现一遍,那意味着 JAVA语言 已经筑基成功了。后面的进阶学习,可以继续跟着我,一起走向架构师之路。

一、题目描述

题目:对象的克隆是Java一项高级技术,可以根据给定的对象,获得与其完全相同的另一个对象。

如果对象的成员变量包含可变引用类型,则需要使用深克隆技术。
你现在会发现使用clone方法来进行克隆,是很麻烦的事。所以还有另一种克隆的方式:序列化克隆

二、解题思路-序列化克隆

创建一个地址类Address

定义三个成员变量表示:国家,省和市。

使用构造方法对它们进行赋值。

并提供对应的get方法和set方法。

重写toString()方法,来输出对象。

再创建一个员工类Employee

定义三个成员变量表示:员工名字,年龄和地址

使用构造方法对它们进行赋值。

并提供对应的get方法和set方法。

重写toString()方法和。

通常情况下,克隆对象都需要使用深克隆。
把对象写入本地文件的方式完成序列化

三、代码详解

地址类:

public class Address implements Serializable 
    private static final long serialVersionUID = 4983187287403615604L;
    private String state; // 表示员工所在的国家
    private String province; // 表示员工所在的省
    private String city; // 表示员工所在的市

    public Address(String state, String province, String city) // 利用构造方法初始化各个域
        this.state = state;
        this.province = province;
        this.city = city;
    

    public String getState() 
        return state;
    

    public void setState(String state) 
        this.state = state;
    

    public String getProvince() 
        return province;
    

    public void setProvince(String province) 
        this.province = province;
    

    public String getCity() 
        return city;
    

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

    public static long getSerialversionuid() 
        return serialVersionUID;
    

    @Override
    public String toString() // 使用地址属性表示地址对象
        StringBuilder sb = new StringBuilder();
        sb.append("国家:" + state + ", ");
        sb.append("省:" + province + ", ");
        sb.append("市:" + city);
        return sb.toString();
    




员工类:

public class Employee implements Serializable 
    private static final long serialVersionUID = 3049633059823371192L;
    private String name; // 表示员工的姓名
    private int age; // 表示员工的年龄
    private Address address;// 表示员工的地址

    public Employee(String name, int age, Address address) // 利用构造方法初始化各个域
        this.name = name;
        this.age = age;
        this.address = address;
    
    
     public Employee() // 利用构造方法初始化各个域
        super();
    

    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;
    

    public Address getAddress() 
        return address;
    

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

    @Override
    public String toString() // 重写toString()方法
        StringBuilder sb = new StringBuilder();
        sb.append("姓名:" + name + ", ");
        sb.append("年龄:" + age + "\\n");
        sb.append("地址:" + address);
        return sb.toString();
    



测试类

public class Test 
    public static void main(String[] args) 
        System.out.println("序列化之前:");
        Address address = new Address("中国", "吉林", "长春");// 创建address对象
        Employee employee1 = new Employee("小虚竹", 30, address);// 创建employee1对象
        System.out.println("员工1的信息:");
        System.out.println(employee1);// 输出employee1对象
        System.out.println("序列化之后:");
        ObjectOutputStream out = null;
        ObjectInputStream in = null;
        Employee employee2 = null;
        try 
            out = new ObjectOutputStream(new FileOutputStream("employee.dat"));
            out.writeObject(employee1);// 将对象写入到本地文件中
            in = new ObjectInputStream(new FileInputStream("employee.dat"));
            employee2 = (Employee) in.readObject();// 从本地文件读取对象
         catch (FileNotFoundException e) 
            e.printStackTrace();
         catch (IOException e) 
            e.printStackTrace();
         catch (ClassNotFoundException e) 
            e.printStackTrace();
         finally 
            if (in != null) 
                try 
                    in.close();// 关闭输入流
                 catch (IOException e) 
                    e.printStackTrace();
                
            
            if (out != null) 
                try 
                    out.close();// 关闭输出流
                 catch (IOException e) 
                    e.printStackTrace();
                
            
        

        if (employee2 != null) 
            employee2.getAddress().setState("中国"); // 修改员工地址
            employee2.getAddress().setProvince("四川"); // 修改员工地址
            employee2.getAddress().setCity("成都"); // 修改员工地址
            employee2.setName("大虚竹"); // 修改员工名字
            employee2.setAge(24);// 修改员工年龄
            System.out.println("员工1的信息:");
            System.out.println(employee1);// 输出employee1对象
            System.out.println("员工2的信息:");
            System.out.println(employee2);// 输出employee2对象
        

    



解题思路二:把对象写入内存,完成序列化

使用ByteArrayInputStream和ByteArrayOutputStream 两个基于内存的流

注:

  • 调用ByteArrayInputStream或ByteArrayOutputStream对象的close方法没有任何意义
  • 这两个基于内存的流只要垃圾回收器清理对象就能够释放资源,这一点不同于对外部资源(如文件流)的释放

代码详解

public class Test2 
    public static void main(String[] args) 
        System.out.println("序列化之前:");
        Address address = new Address("中国", "吉林", "长春");// 创建address对象
        Employee employee1 = new Employee("小虚竹", 30, address);// 创建employee1对象
        System.out.println("员工1的信息:");
        System.out.println(employee1);// 输出employee1对象
        System.out.println("序列化之后:");
       
        Employee employee2 = null;
        try 
            ByteArrayOutputStream bout = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bout);
            oos.writeObject(employee1);
            ByteArrayInputStream bin = new ByteArrayInputStream(bout.toByteArray());
            ObjectInputStream ois = new ObjectInputStream(bin);
            employee2 = (Employee) ois.readObject();// 从内存读取对象
         catch (Exception e) 
            e.printStackTrace();
         

        if (employee2 != null) 
            employee2.getAddress().setState("中国"); // 修改员工地址
            employee2.getAddress().setProvince("四川"); // 修改员工地址
            employee2.getAddress().setCity("成都"); // 修改员工地址
            employee2.setName("大虚竹"); // 修改员工名字
            employee2.setAge(24);// 修改员工年龄
            System.out.println("员工1的信息:");
            System.out.println(employee1);// 输出employee1对象
            System.out.println("员工2的信息:");
            System.out.println(employee2);// 输出employee2对象
        

    



解题思路三:引用springframework

springframework有一个工具方法,常用于对象的克隆

BeanUtils.copyProperties(employee1,employee2);

注意:BeanUtils.copyProperties 方法本质上只是浅克隆,对于引用类型的参数是无法克隆的,要额外处理

代码详解

pom引入

 <properties>
        <org.springframework.version>4.3.25.RELEASE</org.springframework.version>   
    </properties>
    
    <dependencies>

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>$org.springframework.version</version>
        </dependency>
    </dependencies>

测试类

public class Test3 
    public static void main(String[] args) 
        System.out.println("序列化之前:");
        Address address = new Address("中国", "吉林", "长春");// 创建address对象
        Employee employee1 = new Employee("小虚竹", 30, address);// 创建employee1对象
        System.out.println("员工1的信息:");
        System.out.println(employee1);// 输出employee1对象
        System.out.println("序列化之后:");
       
        Employee employee2 = new Employee();
        BeanUtils.copyProperties(employee1,employee2);
        employee2.setAddress(new Address("中国","四川","成都"));
            employee2.setName("大虚竹"); // 修改员工名字
            //employee2.setAge(24);// 修改员工年龄
            System.out.println("员工1的信息:");
            System.out.println(employee1);// 输出employee1对象
            System.out.println("员工2的信息:");
            System.out.println(employee2);// 输出employee2对象
    



四、推荐专栏

《JAVA从零到壹》

《JAVA从零到壹》第四讲:类与对象基础

第六讲:数组包及访问控制

《JAVA从零到壹》第七讲:面向对象高级特性

五、示例源码下载

关注下面的公众号,回复筑基+题目号

筑基35

以上是关于第35题JAVA高级技术-对象克隆4(序列化与对象克隆)的主要内容,如果未能解决你的问题,请参考以下文章

第36题JAVA高级技术-对象克隆5(几种深克隆效率比较)

第33题JAVA高级技术-对象克隆2(浅克隆)

第32题JAVA高级技术-对象克隆1(假克隆)

第34题JAVA高级技术-对象克隆3(深克隆)

JAVA面试题:对象拷贝

对象的创建与克隆