廖雪峰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以后,通 ...
随机推荐
- Oracle数据库联机重定义讲解及错误处理
1.1. 关键字:联机重定义/SYNC_INTERIM_TABLE/GATHER_TABLE_STATS 1.2. 需求:数据表的清理机制需要优化 离线消息表采用delete的方式定期对过期的数据进行 ...
- Qt使用std::sort进行排序
参考: https://blog.csdn.net/u013346007/article/details/81877755 https://www.linuxidc.com/Linux/2017-01 ...
- mysql中关于关联索引的问题——对a,b,c三个字段建立联合索引,那么查询时使用其中的2个作为查询条件,是否还会走索引?
情况描述:在MySQL的user表中,对a,b,c三个字段建立联合索引,那么查询时使用其中的2个作为查询条件,是否还会走索引? 根据查询字段的位置不同来决定,如查询a, a,b a,b, ...
- decltype typename
decltype((variable))总是引用类型,但是decltype(variable)只有当variable是引用类型时才是引用类型. #include <iostream> #i ...
- day34-python操作redis三
Hash类型操作 Hash类型操作 Redis在内存中存储hash类型是以name对应一个字典形式存储的 hset(name,key,value) #name对应的hash中设置一个键值对(不存在,则 ...
- [Codeforces771E]Bear and Rectangle Strips
Problem 给你一个2*n的矩阵,要求你用补充叠的矩阵去框,要求每个矩阵框中的数之和为0,问最多可以用几个矩阵. Solution 首先预处理出一个点到离它最近的一段和为0的区间的左端点 然后到这 ...
- webApp总结
有关Meta 基本Meta <!-- 设置缩放 --> <meta name="viewport" content="width=device-widt ...
- 50个常用的Linux命令(三)基础实例
ls ls -als -l == llls -Aldrwxrwxrwx. 2 root root 6 Dec 21 20:38 Videos-rwxrwxrwx 1 root ...
- xenserver挂载新硬盘
注意:新加硬盘请不要加入raid,否则不认盘 一: 1.1:查看磁盘列表 fdisk -l [root@xenserver zz]# fdisk -l Disk /dev/sdb: 7999.4 GB ...
- box-sizing的用法
默认情况下设置盒子的width是指内容区域,所以在设置边框会使得盒子往外扩张,如果要让css设置的width就是盒子最终的宽度,那么就要设置box-sizing:border-box, ...