#我在头条搞创作第二期#
❝
❤️作者简介:大家好,我是小虚竹。Java领域优质创作者,CSDN博客专家,华为云享专家,掘金年度人气作者,阿里云专家博主,51CTO专家博主
❤️技术活,该赏
❤️点赞 收藏 ⭐再看,养成习惯
❞
零、前言
今天是学习 「JAVA语言」 打卡的第35天,我的学习策略很简单,题海策略+ 费曼学习法。如果能把这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对象
}
}
如图
解题思路四:引用使用kryo序列化
Kryo 序列化:基于Java的快速高效的对象序列化框架,旨在提供快速、高效和易用的API。
Kryo的优势是序列化后size小且速度快,代码简单,号称 Java 中最快的序列化框架。
创建一个SerializationKryoUtils工具类
提供serializationObject序列化对象方法
提供deserialization反序列化对象方法
支持从对象--》字节--》对象的克隆。
代码详解
pom引入
<!-- 使用kryo序列化-->
<dependency>
<groupId>com.esotericsoftware</groupId>
<artifactId>kryo</artifactId>
<version>4.0.2</version>
</dependency>
<dependency>
<groupId>de.javakaffee</groupId>
<artifactId>kryo-serializers</artifactId>
<version>0.45</version>
</dependency>
SerializationKryoUtils
package com.xiaoxuzhu;
import de.javakaffee.kryoserializers.UnmodifiableCollectionsSerializer;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.Serializable;
import org.objenesis.strategy.StdInstantiatorStrategy;
import com.esotericsoftware.kryo.Kryo;
import com.esotericsoftware.kryo.io.Input;
import com.esotericsoftware.kryo.io.Output;
/**
* Description: 使用kryo序列化工具包
*
* @author xiaoxuzhu
* @version 1.0
*
* <pre>
* 修改记录:
* 修改后版本 修改人 修改日期 修改内容
* 2022/4/19.1 xiaoxuzhu 2022/4/19 Create
* </pre>
* @date 2022/4/19
*/
public class SerializationKryoUtils {
private static final ThreadLocal<Kryo> kryoThreadLocal = new ThreadLocal<Kryo>() {
@Override
protected Kryo initialValue() {
Kryo kryo = new Kryo();
UnmodifiableCollectionsSerializer.registerSerializers(kryo);
kryo.setRegistrationRequired(false);
kryo.setInstantiatorStrategy(new Kryo.DefaultInstantiatorStrategy(new StdInstantiatorStrategy()));
return kryo;
}
};
public static Kryo getInstance() {
return kryoThreadLocal.get();
}
/**
* 序列化对象
*
* @param source 对象
* @param <T>
* @return
*/
public static <T extends Serializable> byte[] serializationObject(T source) {
Kryo kryo = getInstance();
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
Output output = new Output(byteArrayOutputStream);
try {
kryo.writeClassAndObject(output, source);
} catch (Exception e) {
e.printStackTrace();
} finally {
kryoThreadLocal.remove();
}
output.flush();
output.close();
byte[] bytes = byteArrayOutputStream.toByteArray();
try {
byteArrayOutputStream.flush();
byteArrayOutputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
return bytes;
}
/**
* 反序列化对象
*
* @param source
* @param <T>
* @return
*/
@SuppressWarnings("unchecked")
public static <T> T deserializationObject(byte[] source) {
Kryo kryo = getInstance();
try {
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(source);
Input input = new Input(byteArrayInputStream);
return (T) kryo.readClassAndObject(input);
} catch (Exception e) {
e.printStackTrace();
} finally {
kryoThreadLocal.remove();
}
return null;
}
}
测试类:
package com.xiaoxuzhu;
/**
* Description: 测试类
*
* @author xiaoxuzhu
* @version 1.0
*
* <pre>
* 修改记录:
* 修改后版本 修改人 修改日期 修改内容
* 2022/4/19.1 xiaoxuzhu 2022/4/19 Create
* </pre>
* @date 2022/4/19
*/
public class Test4 {
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;
byte[] bytes = SerializationKryoUtils.serializationObject(employee1);
employee2 = SerializationKryoUtils.deserializationObject(bytes);
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对象
}
}
我是虚竹哥,我们下一题见~