Java序列化Serializable
1、什么是序列化和反序列化
Serialization(序列化)是一种将对象以一连串的字节描述的过程;deserialization(反序列化)是一种将这些字节重建成一个对象的过程。
2、什么情况下需要序列化
a)当你想把内存中的对象保存到一个文件 中或者数据库中的时候;
b)当你想用套接字socket在网络上传送对象的时候;
c)当你想通过RMI(Remote Method Invocation)传输对象的时候;
3、如何实现序列化
将需要序列化的类实现Serializable接口就可以了,Serializable接口中没有任何方法,可以理解为一个标记,即表明这个类可以序列化。
4、序列化和反序列化例子
如果我们想要序列化一个对象,首先要创建某些OutputStream(如FileOutputStream、ByteArrayOutputStream等),然后将这些OutputStream封装在一个ObjectOutputStream中。这时候,只需要调用writeObject()方法就可以将对象序列化,并将其发送给OutputStream(记住:对象的序列化是基于字节的,不能使用Reader和Writer等基于字符的层次结构)。而反序列的过程(即将一个序列还原成为一个对象),需要将一个InputStream(如FileInputStream、ByteArrayInputStream等)封装在ObjectInputStream内,然后调用readObject()即可。
- /**
- * Description 序列化接口Serializable示例
- * Date 2018-10-22
- * @author Encore.M
- */
- package com.meng.javalanguage.Serializable;
- import java.io.FileInputStream;
- import java.io.FileNotFoundException;
- import java.io.FileOutputStream;
- import java.io.IOException;
- import java.io.ObjectInputStream;
- import java.io.ObjectOutputStream;
- import java.io.Serializable;
- public class MyTest implements Serializable {
- private static final long serialVersionUID = 1L;
- private String name = "mengjia";
- private int age = 24;
- public static void main(String[] args) {
- // 以下代码实现序列化
- try{
- // 输出流保存的文件名为my.out
- // ObjectOutputStream能把Object输出成Byte流
- ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("my.out"));
- MyTest myTest = new MyTest();
- oos.writeObject(myTest);
- oos.flush(); // 缓冲流
- oos.close(); // 关闭流
- }catch(FileNotFoundException e) {
- e.printStackTrace();
- }catch(IOException e) {
- e.printStackTrace();
- }
- deserialization(); // 调用下面的反序列化函数
- }
- /**
- * @description 反序列化,将字节流重建为对象
- * @param 无
- * @return 无
- * @author Encore.M
- * @date 2018-10-22
- * @annotation 需捕获ClassNotFoundException和IOException异常或在函数定义时throws这两个异常
- */
- public static void deserialization() {
- ObjectInputStream oin = null; // 局部变量必须要初始化
- try{
- oin = new ObjectInputStream(new FileInputStream("my.out"));
- }catch(FileNotFoundException e1) {
- e1.printStackTrace();
- }catch(IOException e1 ) {
- e1.printStackTrace();
- }
- MyTest mts = null;
- try{
- mts = (MyTest)oin.readObject(); // 由Object对象向下转型为MyTest对象
- }catch(ClassNotFoundException e) {
- e.printStackTrace();
- }catch(IOException e) {
- e.printStackTrace();
- }
- System.out.println("name = " + mts.name);
- System.out.println("age = " + mts.age);
- }
- }
序列化MyTest成功后会在此项目的工作空间生成一个my.out文件。而反序列化MyTest是读取my.out后生成一个MyTest对象。
5、serialVersionUID
注意到上面程序中有一个serialVersionUID,实现了Serializable接口之后,Eclipse就会提示你增加一个serialVersionUID,虽然不加的话上述程序依然能够正常运行。
序列化ID在Eclipse下提供了两种生成策略
- 一个是固定的1L
- 一个是随机生成的一个不重复的long类型数据(实际上是使用JDK工具,根据类名、接口名、成员方法及属性等来生成)
上面程序中,输出对象和读入对象使用的是同一个MyTest类。
如果是通过网络传输的话,当MyTest类的serialVersionUID不一致时,那么反序列化就不能正常进行。例如在客户端A中MyTest类的serialVersionUID=1L,而在客户端B中MyTest类的serialVersionUID=2L,那么就不能重构这个MyTest对象。视图重构则会报java.io.InvalidClassException异常,因为这两个类的版本不一致。
如果没有特殊需求的话,使用默认的1L就可以,这样可以确保代码一致时反序列化成功。那么随机生成的序列化ID有什么用呢,有些时候,通过改变序列化ID可以用来限制某些用户的使用。
6、静态变量序列化
串行化只能保存对象的非静态成员变量,不能保存任何的成员方法和静态的成员变量,而且串行化保存的只是变量的值,对于变量的任何修饰符都不能保存。
如果把MyTest类中的name定义为static类型的话,试图重构,就不能得到原来的值,只能得到null。说明对静态成员变量值是不保存的。这其实比较容易理解,序列化保存的是对象的状态,静态变量属于类的状态,因此,序列化并不保存静态变量。
7、transient关键字
transient关键字的作用是阻止实例中那些用此关键字声明的变量的持久化;当对象被反序列化时(从源码读取字节序列进行重构),这样的实例变量值不会被恢复。
当某些变量不想被序列化,同时又不适合使用static关键字声明时,可以用transient关键字来声明该变量。在被反序列化后,transient变量的值被设为初始值,如int型的默认是0,对象型的是null。
注:对于某些类型的属性,其状态是瞬时的,这样的属性是无法保存其状态的。例如一个线程属性或是需要访问IO、本地资源、网络资源等的属性,对于这些字段,我们必须用transient关键字标明,否则编译器将报错。
8、序列化中的继承问题
1)当一个父类实现序列化,子类自动实现序列化,不需要显示实现Serializable接口。
2)一个子类实现了Serializable接口,它的父类都没有实现Serializable接口,要想将父类对象也序列化,就需要让父类也实现Serializable接口。
第二种情况中,如果父类不实现Serializable接口的话,就需要有默认的无参构造函数。在父类没有实现Serializable接口时,虚拟机是不会序列化父对象的,而一个Java对象的构造必须先有父对象,才有子对象,反序列化也不例外。所以反序列化时,为了构造父对象,只能调用父类的无参构造函数作为默认的父对象。因此当我们取父对象的变量值时,它的值是调用父类无参构造函数后的值。如果你考虑到这种序列化的情况,在父类无参构造函数中对变量进行初始化,否则,父类变量值都是默认声明的值,如int型的默认是0,string型的默认是null。
9、总结
序列化给我们提供了一种技术,用户保存对象的变量,以便于 传输。虽然也可以使用别的一些方法实现同样的功能,但是java给我们提供的方法使用起来是非常方便的。
Java序列化Serializable的更多相关文章
- Java 序列化Serializable详解
Java 序列化Serializable详解(附详细例子) Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化Serialization(序列化)是一种将对象以一连 ...
- Java 序列化Serializable详解(附详细例子)
Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化 Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization ...
- Java 序列化Serializable具体解释(附具体样例)
Java 序列化Serializable具体解释(附具体样例) 1.什么是序列化和反序列化 Serialization(序列化)是一种将对象以一连串的字节描写叙述的过程:反序列化deserializa ...
- java 序列化Serializable 详解
Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization是 ...
- Java 序列化Serializable详解(附详细例子)
Java 序列化Serializable详解(附详细例子) 1.什么是序列化和反序列化Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization是 ...
- java:序列化Serializable 接口
java:序列化Serializable 接口 public class SerializePerson implements Serializable { private String name; ...
- Java 序列化Serializable接口
1 什么是序列化和反序列化 Serialization(序列化)是一种将对象以一连串的字节描述的过程:反序列化deserialization是一种将这些字节重建成一个对象的过程. 2 什么情况下需要 ...
- Java序列化Serializable和Externalizable
纸上得来终觉浅,绝知此事要躬行 --陆游 问渠那得清如许,为有源头活水来 --朱熹 什么是Java序列化?为什么出现Java序列化?怎样实现Java序列化? 一.什么是Java序列化 ...
- Java 序列化Serializable
a)当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口: b)当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化: c) static,trans ...
随机推荐
- C++回顾day01---<C++对C的增强>
一:命名空间 二:三目运算符 (一)C语言三目运算符返回值(不可修改) (二)C++中三目运算符直接返回变量本身(可以直接进行修改) 三:C++要求所有函数必须有类型(不重要) (一)C语言允许函数无 ...
- Kafka权威指南 读书笔记之(三)Kafka 生产者一一向 Kafka 写入数据
不管是把 Kafka 作为消息队列.消息总线还是数据存储平台来使用 ,总是需要有一个可以往 Kafka 写入数据的生产者和一个从 Kafka 读取数据的消费者,或者一个兼具两种角色的应用程序. 开发者 ...
- Linux记录-I/O系统监控
几个基本的概念 在研究磁盘性能之前我们必须先了解磁盘的结构,以及工作原理.不过在这里就不再重复说明了,关系硬盘结构和工作原理的信息可以参考维基百科上面的相关词条——Hard disk drive(英文 ...
- HDU 1021(斐波那契数与因子3 **)
题意是说在给定的一种满足每一项等于前两项之和的数列中,判断第 n 项的数字是否为 3 的倍数. 斐波那契数在到第四十多位的时候就会超出 int 存储范围,但是题目问的是是否为 3 的倍数,也就是模 3 ...
- 伯克利SocketAPI(一) socket的C语言接口/最简单的服务器和对应的客户端C语言实现
1. 头文件 2. API函数 3. 最简单的服务器和对应的客户端C语言实现 3.1 server #include <sys/types.h> #include <sys/sock ...
- 020、搭建本地Registry(2019-01-11 周五)
参考https://www.cnblogs.com/CloudMan6/p/6902325.html Docker Hub 虽然方便,但还是有些限制,比如 1.需要Internet连接,上 ...
- silverlight用Encoding.UTF8读取shape文件的中文属性值 出现乱码
最近用Silverlight读取shape文件,读出的属性居然是乱码. 原因是:Silverlight不支持GB2312. 解决方案: 下载该地址的代码http://encoding4silverli ...
- linux大法好。。。。。
vim: 移动光标至段首:^或者home键 移动光标至段尾:$或者end键 删除光标位置到本行开头:d0或者d^ 删除光标位置到本行末尾:D或者d$ 撤销操作:u 取消撤销操作:ctrl+r ---- ...
- SQL Server进阶 索引
create unique index 和create index 的区别? CREATE UNIQUE INDEX ProviderInfo_Id_uindex ON dbo.ProviderInf ...
- JS处理数据四舍五入
一,使用Math.toFixed toFixed() 方法可把 Number 四舍五入为指定小数位数的数字. 语法NumberObject.toFixed(num) 但是网友说toFixed bug比 ...