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序列化的更多相关文章

  1. 廖雪峰Java6 IO编程-2input和output-4Filter模式

    1.JDK提供的InputStream分为两类: 直接提供数据的InputStream * FileInputStream:从文件读取 * ServletInputStream:从HTTP请求读取数据 ...

  2. 廖雪峰Java6 IO编程-2input和output-5操作zip

    1.ZipInputStream是一种FilterInputStream 可以直接读取zip的内容 InputStream->FilterInputStream->InflateInput ...

  3. 廖雪峰Java6 IO编程-2input和output-6classpath资源

    1.从classpath读取文件可以避免不同环境下文件路径不一致的问题. Windows和Linux关于路径的表示不一致 Windows:C:\conf\default.properties Linu ...

  4. 廖雪峰Java6 IO编程-3Reader和Writer-2Writer

    1.java.io.Writer和java.io.OutputStream的区别 OutputStream Writer 字节流,以byte为单位 字符流,以char为单位 写入字节(0-255):v ...

  5. 廖雪峰Java6 IO编程-3Reader和Writer-1Reader

    1.java.io.Reader和java.io.InputStream的区别 InputStream Reader 字节流,以byte为单位 字符流,以char为单位 读取字节(-1,0-255): ...

  6. 廖雪峰Java13网络编程-1Socket编程-2TCP编程

    1. Socket 在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据 ...

  7. 廖雪峰Java13网络编程-3其他-1HTTP编程

    1.HTTP协议: Hyper Text Transfer Protocol:超文本传输协议 基于TCP协议之上的请求/响应协议 目前使用最广泛的高级协议 * 使用浏览器浏览网页和服务器交互使用的就是 ...

  8. 廖雪峰Java13网络编程-1Socket编程-5UDP编程

    1. UDP编程: 不需要建立连接 可以直接发送和接收数据 1.1 客户端 DatagramSocket sock = new DatagramSocket(){} sock.connect(addr ...

  9. 廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程

    TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通 ...

随机推荐

  1. Oracle数据库联机重定义讲解及错误处理

    1.1. 关键字:联机重定义/SYNC_INTERIM_TABLE/GATHER_TABLE_STATS 1.2. 需求:数据表的清理机制需要优化 离线消息表采用delete的方式定期对过期的数据进行 ...

  2. Qt使用std::sort进行排序

    参考: https://blog.csdn.net/u013346007/article/details/81877755 https://www.linuxidc.com/Linux/2017-01 ...

  3. mysql中关于关联索引的问题——对a,b,c三个字段建立联合索引,那么查询时使用其中的2个作为查询条件,是否还会走索引?

    情况描述:在MySQL的user表中,对a,b,c三个字段建立联合索引,那么查询时使用其中的2个作为查询条件,是否还会走索引? 根据查询字段的位置不同来决定,如查询a,     a,b    a,b, ...

  4. decltype typename

    decltype((variable))总是引用类型,但是decltype(variable)只有当variable是引用类型时才是引用类型. #include <iostream> #i ...

  5. day34-python操作redis三

    Hash类型操作 Hash类型操作 Redis在内存中存储hash类型是以name对应一个字典形式存储的 hset(name,key,value) #name对应的hash中设置一个键值对(不存在,则 ...

  6. [Codeforces771E]Bear and Rectangle Strips

    Problem 给你一个2*n的矩阵,要求你用补充叠的矩阵去框,要求每个矩阵框中的数之和为0,问最多可以用几个矩阵. Solution 首先预处理出一个点到离它最近的一段和为0的区间的左端点 然后到这 ...

  7. webApp总结

    有关Meta 基本Meta <!-- 设置缩放 --> <meta name="viewport" content="width=device-widt ...

  8. 50个常用的Linux命令(三)基础实例

    ls ls -als -l == llls -Aldrwxrwxrwx.  2 root   root       6 Dec 21 20:38 Videos-rwxrwxrwx   1 root   ...

  9. xenserver挂载新硬盘

    注意:新加硬盘请不要加入raid,否则不认盘 一: 1.1:查看磁盘列表 fdisk -l [root@xenserver zz]# fdisk -l Disk /dev/sdb: 7999.4 GB ...

  10. box-sizing的用法

        默认情况下设置盒子的width是指内容区域,所以在设置边框会使得盒子往外扩张,如果要让css设置的width就是盒子最终的宽度,那么就要设置box-sizing:border-box,     ...