第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对象
四、推荐专栏
五、示例源码下载
关注下面的公众号,回复筑基+题目号
筑基35
以上是关于第35题JAVA高级技术-对象克隆4(序列化与对象克隆)的主要内容,如果未能解决你的问题,请参考以下文章