廖雪峰Java6 IO编程-2input和output-7序列化
1.序列化
序列化是指把一个Java对象变成二进制内容byte[]
- 序列化后可以把byte[]保存到文件中
- 序列化后可以把byte[]通过网络传输
- 一个Java对象要能序列化,必须实现Serializable接口:
* Serializable接口没有定义任何方法
* 空接口被称为标记接口(Marker Interface) - ObjectOutputStream负责把一个Java对象写入二进制流:
try(ObjectOutputStream output = new ObjectOutputStream(...)){
output.writeObject(new Person("小明"));
output.writeObject(new Person("小红"));
}
2.反序列化
反序列化是指把一个二进制内容(byte[])变成Java对象
- 反序列化后可以从文件读取byte[]并变成Java对象
- 反序列化后可以从网络读取byte[]并变为Java对象
- ObjectInputStream负责从二进制流读取一个Java对象
try(ObjectInputStream input = new ObjectInputStream(...)){
Object p1 = input.readObject();//使用readObject()获取Object对象
Person p2 = (Person) input.readObject();//如果读取的Person类型,需要强制转型
}
readObject()可能抛出的异常:
- ClassNotFoundException:没有找到对应的Class。反序列化的Java程序没有Person类,不能反序列化
- InvalidClassException:Class不匹配。序列化的Person的age是int,反序列化的Person的age是long类型

反序列化的重要特点: - 反序列化由JVM直接构造出Java对西那个,不调用构造方法,即构造方法的初始化代码根本不执行。
3.示例
Person.java
import java.io.Serializable;
public class Person implements Serializable {
private String name;
public Person(String name){
System.out.println(name+"诞生了");
this.name = name;
}
public String getName(){
return name;
}
public void setName(String name){
this.name = name;
}
}
Main.java
public class Main {
public static void main(String[] args) throws IOException,ClassNotFoundException {
String dataFile = "saved.data";
try(ObjectOutputStream output = new ObjectOutputStream(
new BufferedOutputStream(
new FileOutputStream(dataFile)))){
//依次写入int,String,Person对象
output.writeInt(999);
output.writeUTF("Hello world");
output.writeObject(new Person("小明"));
output.writeObject(new Person("小红"));
}
System.out.println("Read...");
try(ObjectInputStream input = new ObjectInputStream(
new BufferedInputStream(
new FileInputStream(dataFile)))){
//依次读取int,String,Person对象
System.out.println(input.readInt());
System.out.println(input.readUTF());
Person p1 = (Person) input.readObject();//构造方法不会执行
System.out.println(p1);
Person p2 = (Person) input.readObject();
System.out.println(p2);
}
}
}

## 4. serialVersionUID
为Person添加serialVersionUID,Main操作1次。
Person中serialVersionUID发生改变,只保留读取,再次执行Main,结果报错
Person.java
```#java
import java.io.Serializable;
public class Person implements Serializable {
public static final long serialVersionUID = 1234567890123456L;//Main执行1次后将UID修改为1234567890L
private String name;
public Person(String name){
System.out.println(name+"诞生了");
this.name = name;
}
public String getName(){
return name;
}
public void setName(String name){
this.name = name;
}
}
```#java
public class Main {
public static void main(String[] args) throws IOException,ClassNotFoundException {
String dataFile = "saved.data";
//修改UID后,注释写入代码块
try(ObjectOutputStream output = new ObjectOutputStream(
new BufferedOutputStream(
new FileOutputStream(dataFile)))){
//依次写入int,String,Person对象
output.writeInt(999);
output.writeUTF("Hello world");
output.writeObject(new Person("小明"));
output.writeObject(new Person("小红"));
}
System.out.println("Read...");
try(ObjectInputStream input = new ObjectInputStream(
new BufferedInputStream(
new FileInputStream(dataFile)))){
//依次读取int,String,Person对象
System.out.println(input.readInt());
System.out.println(input.readUTF());
Person p1 = (Person) input.readObject();//构造方法不会执行
System.out.println(p1);
Person p2 = (Person) input.readObject();
System.out.println(p2);
}
}
}

5.总结:
- Java序列化:指将对象转为字节序列的过程
- Java反序列化:将字节序列转换为目标对象的过程

淘宝购买布衣柜,卖家将所有的组件打包到纸箱中,买家自己组装。
* 序列化:把柜子拆散,并装到箱子中
* 反序列化:买家收到箱子后,把这些配件组装成桌子。 - 可序列化的Java对象必须实现java.io.Serializable接口
- 类似Serializable这样的空接口被称为标记接口(Marker Interface)
- 反序列化时不调用构造方法
- 可设置serialVersionUID作为版本号(非必需)
- Java的序列化机制仅适用于Java,如果需要与其他语言交换数据,必须使用通用的序列化方法,例如JSON。
廖雪峰Java6 IO编程-2input和output-7序列化的更多相关文章
- 廖雪峰Java6 IO编程-2input和output-4Filter模式
1.JDK提供的InputStream分为两类: 直接提供数据的InputStream * FileInputStream:从文件读取 * ServletInputStream:从HTTP请求读取数据 ...
- 廖雪峰Java6 IO编程-2input和output-5操作zip
1.ZipInputStream是一种FilterInputStream 可以直接读取zip的内容 InputStream->FilterInputStream->InflateInput ...
- 廖雪峰Java6 IO编程-2input和output-6classpath资源
1.从classpath读取文件可以避免不同环境下文件路径不一致的问题. Windows和Linux关于路径的表示不一致 Windows:C:\conf\default.properties Linu ...
- 廖雪峰Java6 IO编程-3Reader和Writer-2Writer
1.java.io.Writer和java.io.OutputStream的区别 OutputStream Writer 字节流,以byte为单位 字符流,以char为单位 写入字节(0-255):v ...
- 廖雪峰Java6 IO编程-3Reader和Writer-1Reader
1.java.io.Reader和java.io.InputStream的区别 InputStream Reader 字节流,以byte为单位 字符流,以char为单位 读取字节(-1,0-255): ...
- 廖雪峰Java13网络编程-1Socket编程-2TCP编程
1. Socket 在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据 ...
- 廖雪峰Java13网络编程-3其他-1HTTP编程
1.HTTP协议: Hyper Text Transfer Protocol:超文本传输协议 基于TCP协议之上的请求/响应协议 目前使用最广泛的高级协议 * 使用浏览器浏览网页和服务器交互使用的就是 ...
- 廖雪峰Java13网络编程-1Socket编程-5UDP编程
1. UDP编程: 不需要建立连接 可以直接发送和接收数据 1.1 客户端 DatagramSocket sock = new DatagramSocket(){} sock.connect(addr ...
- 廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程
TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通 ...
随机推荐
- 关于antd 日期组件只选择年份,设置mode=year无法获取value的解决办法
antd3.0后的某个版本后终于支持了只选择年份的设置.当时2.x版本的时候还不支持只选择年份,我们项目中有这个只选择年份的需求,为了ui风格的一致,只好自己撸了一个. 如今真是普天同庆!
- pandas中的空值处理
1.空值 1.1 有两种丢失数据: None: Python自带的数据类型 不能参与到任何计算中 np.nan: float类型 能参与计算,但结果总是nan # None+2 # 报错 # np.n ...
- Python ORM框架之SQLAlchemy
前言: Django的ORM虽然强大,但是毕竟局限在Django,而SQLAlchemy是Python中的ORM框架: SQLAlchemy的作用是:类/对象--->SQL语句--->通过 ...
- oracle中日期类型 to_date 和to_timestamp什么区别啊?
1.to_date() 和to_timestamp()区别 由于oracle中date类型只支持到秒,不支持到毫秒,所以to_date()不能取到毫秒.如果要取到毫秒,oracle 9i以上版本,可以 ...
- MySQL 用户连接与用户线程
本文转载自公众号数据库随笔,作者happypig 微信看起来麻烦 pig已经好长一段时间没有分享文章了,有点对不起订阅的朋友.最近在做比较复杂跟困难的事情,也并不一定最终会有成果,因此必须对此沉默. ...
- JS实现大整数乘法(性能优化、正负整数)
本方法的思路为: 一:检查了输入的合法性(非空,无非法字符) 二:检查输入是否可以进行简单计算(一个数为 0,1,+1,-1) 三:去掉输入最前面可能有的正负符号,并判断输出的正负 四:将输入的值分成 ...
- poj1873(枚举+凸包)
The Fortified Forest Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 7291 Accepted: 2 ...
- 谷歌浏览器chrome的vuejs devtools 插件的安装
(推荐方法2) 安装方法1: 需正常打开chrome商店,搜索vuejs devtools 安装.chrome://extensions/ 开发者工具-扩展程序下启用: 方法2: github下载插件 ...
- Linux分区知识及企业场景分区76
文件系统就相当于装修一样.这个硬盘拿过来了,分完区了,没有格式化. 没有格式化就相当于没有装修.[分区]不是必须的. 如果没有文件系统就不能放数据,文件系统可以理解为一个软件, 它的实现形式是软件,这 ...
- spring集成jedis简单实例
jedis是redis的java客户端,spring将redis连接池作为一个bean配置. redis连接池分为两种,一种是“redis.clients.jedis.ShardedJedisPool ...