Java 序列化的缺点
Java 提供的对象输入流(ObjectInputStream)和输出流(ObjectOutputStream),可以直接把 Java 对象作为可存储的字节数据写入文件,也可以传输到网络上。对于程序员来说,基于 JDK 默认的序列化机制可以避免操作底层的字节数组,从而提高开发效率。Java 序列化的主要目的是网络传输和对象持久化。
一、无法跨语言
无法跨语言,是 Java 序列化最致命的问题。对于跨进程的服务调用,服务提供者可能是 Java 意外的其他语言,当我们需要和异构语言交互时,Java 序列化就难以胜任
由于 Java 序列化技术是 Java 语言内部的私有协议,其他语言并不支持,对于用户来说它完全是个黑盒子。对于 Java 序列化后的字节数组,别的语言无法进行反序列化,这就严重阻碍了它的应用。
事实上,目前几乎所有流行的 Java RCP 通信框架,都没有使用 Java 序列化作为编解码框架,原因就在于它无法跨语言,而这些 RPC 框架往往需要支持跨语言调用。
二、序列化后的码流太大
【1】通过一个实例看下 Java 序列化后的字节数组大小。如下 UserInfo 对象是实现了序列化接口的对象,并生成了默认的序列号:serialVersionUID = 1L。说明 UserInfo 对象可以通过 JDK 默认的序列化机制进行序列化和反序列化。并创建了一个与之比较码流大小的方法 codeC ,此方法基于 ByteBuffer 的通用二进制编解码技术对 UserInfo 对象进行编码,编码结果也是 byte 数组。
【序列化 ID 问题】:两个客户端 A 和 B 试图通过网络传递对象数据,A 端将对象 C 序列化为二进制数据再传给 B,B 反序列化得到 C。
问题:C 对象的全类路径假设为 com.yintong.UserInfo,在 A 和 B 端都有这么一个类文件,功能代码完全一致。也都实现了 Serializable 接口,但是反序列化时总是提示不成功。
解决:虚拟机是否允许反序列化,不仅取决于类路径和功能代码是否一致,一个非常重要的一点是两个类的序列化 ID 是否一致(就是 private static final long serialVersionUID = 1L)。清单 1 中,虽然两个类的功能代码完全一致,但是序列化 ID 不同,他们无法相互序列化和反序列化。
1 //实现了序列化接口的实例
2 public class UserInfo implements Serializable{
3
4 private static final long serialVersionUID = 1L;
5
6 //用户ID
7 private int ID;
8 //用户名
9 private String name;
10
11 //有参构造器
12 public UserInfo(int iD, String name) {
13 super();
14 ID = iD;
15 this.name = name;
16 }
17
18 //根据 buffer缓冲区 获取字节数组
19 public byte[] codeC() {
20 //定义字节缓冲区
21 ByteBuffer buffer = ByteBuffer.allocate(1024);
22 //获取name属性的二进制字节流
23 byte[] value = this.name.getBytes();
24 //用于写入 int 值的相对 put 方法(可选操作)。
25 //将 n 个包含给定 int 值的字节按照当前的字节顺序写入到此缓冲区的当前位置,然后将该位置增加 n。
26 buffer.putInt(value.length);
27 //存入 二进制值
28 buffer.put(value);
29
30 //写入一个int值=ID到ByteBuffer中。
31 buffer.putInt(this.ID);
32 //切换为读
33 buffer.flip();
34 value = null;
35 //remaining 返回剩余的可用长度,次长度为实际读取的数据长度
36 byte[] result = new byte[buffer.remaining()];
37 //获取buffer中的值到 result
38 buffer.get(result);
39 return result;
40 }
41
42 public int getID() {
43 return ID;
44 }
45
46 public void setID(int iD) {
47 ID = iD;
48 }
49
50 public String getName() {
51 return name;
52 }
53
54 public void setName(String name) {
55 this.name = name;
56 }
57 }
【2】下面是测试程序,先调用两种编码接口对 UserInfo 进行编码,然后分别打印两者编码后的流大小进行对比。
1 public class TestUserInfo {
2
3 public static void main(String[] args) throws Exception {
4 UserInfo info = new UserInfo(101, "zheng zhao xiang");
5 //JDK 序列化流程
6 ByteArrayOutputStream bos = new ByteArrayOutputStream();
7 ObjectOutputStream oos = new ObjectOutputStream(bos);
8 oos.writeObject(info);
9 oos.flush();
10 oos.close();
11 byte[] b = bos.toByteArray();
12 System.out.println("JDB 序列化后流的长度: "+b.length);
13 bos.close();
14 System.out.println("通过缓冲区 buffer 处理后的流长度: "+info.codeC().length);
15 }
16 }
【3】测试结果如下:采用 JDK 序列化机制编码后的二进制数组大小是通过缓冲区处理后的 4 倍。

三、序列化性能太低
【1】下面从序列化的性能角度看下 JDK 的表现如何。将之前的例子进行修改。对 UserInfo 中的 codeC 方法改造如下:
1 public byte[] codeC(ByteBuffer buffer) {
2 //清空缓冲区
3 buffer.clear();
4 //获取name属性的二进制字节流
5 byte[] value = this.name.getBytes();
6 //用于写入 int 值的相对 put 方法(可选操作)。
7 //将 n 个包含给定 int 值的字节按照当前的字节顺序写入到此缓冲区的当前位置,然后将该位置增加 n。
8 buffer.putInt(value.length);
9 //存入 二进制值
10 buffer.put(value);
11
12 //写入一个int值=ID到ByteBuffer中。
13 buffer.putInt(this.ID);
14 //切换为读
15 buffer.flip();
16 value = null;
17 //remaining 返回剩余的可用长度,次长度为实际读取的数据长度
18 byte[] result = new byte[buffer.remaining()];
19 //获取buffer中的值到 result
20 buffer.get(result);
21 return result;
22 }
【2】对 Java 序列化和二进制编码分别进行性能测试,编码 100万次,然后统计消耗的总时间:
1 public class PerformTestUserInfo {
2 public static void main(String[] args) throws IOException {
3 UserInfo info = new UserInfo(101, "zheng zhao xiang");
4 //循环次数
5 //JDK 序列化流程
6 ByteArrayOutputStream bos = null;
7 int loop = 1000000;
8 ObjectOutputStream oos = null;
9 //写之前获取系统时间
10 long startTimeMillis = System.currentTimeMillis();
11 for(int i=0; i<loop; i++) {
12 bos = new ByteArrayOutputStream();
13 oos = new ObjectOutputStream(bos);
14 oos.writeObject(info);
15 oos.flush();
16 oos.close();
17 byte[] b = bos.toByteArray();
18 bos.close();
19 }
20 long endTimeMillis = System.currentTimeMillis();
21 System.out.println("JDK 序列化花费的时间: "+(endTimeMillis - startTimeMillis)+" ms");
22
23 ByteBuffer buffer = ByteBuffer.allocate(1024);
24 startTimeMillis = System.currentTimeMillis();
25 for(int i=0; i<loop; i++) {
26 info.codeC(buffer);
27 }
28 endTimeMillis = System.currentTimeMillis();
29 System.out.println("通过缓冲区 buffer花费的时间: "+(endTimeMillis - startTimeMillis)+" ms");
30 }
31 }
【3】结果展示:结果非常令人惊讶,Java 序列化的性能只有二进制编码的 11% 左右,可见原生序列化的性能很差。

四、结论
无论是序列化后的码流大小,还是序列化的性能,JDK 默认的序列化机制表现都很差。因此,我们通常不会选择 Java 序列化作为远程跨节点调用的编解码框架。而是使用业界提供的很多优秀的编解码框架,它们在克服了 JDK 默认的序列化框架缺点的基础上,还增加了很多亮点。例如:Google 的 Protobuf、Facebook 的 Thrift 和 JBoss 的 Marshalling 等等,我后期都会学习和整理相关的博文。

----架构师资料,关注公众号获取----
Java 序列化的缺点的更多相关文章
- java序列化框架(protobuf、thrift、kryo、fst、fastjson、Jackson、gson、hessian)性能对比
我们为什么要序列化 举个栗子:下雨天我们要打伞,但是之后我们要把伞折叠起来,方便我们存放.那么运用到我们java中道理是一样的,我们要将数据分解成字节流,以便存储在文件中或在网络上传输,这叫序列 ...
- 各种Java序列化性能比较
转载:http://www.jdon.com/concurrent/serialization.html 这里比较Java对象序列化 XML JSON Kryo POF等序列化性能比较. 很多人以 ...
- Java序列化的几种方式以及序列化的作用
Java序列化的几种方式以及序列化的作用 本文着重讲解一下Java序列化的相关内容. 如果对Java序列化感兴趣的同学可以研究一下. 一.Java序列化的作用 有的时候我们想要把一个Java对象 ...
- [java]序列化框架性能对比(kryo、hessian、java、protostuff)
序列化框架性能对比(kryo.hessian.java.protostuff) 简介: 优点 缺点 Kryo 速度快,序列化后体积小 跨语言支持较复杂 Hessian 默认支持跨语言 较慢 Pro ...
- Java序列化技术
Java序列化与反序列化是什么?为什么需要序列化与反序列化?如何实现Java序列化与反序列化? Java序列化是指把Java对象转换为字节序列的过程:而Java反序列化是指把字节序列恢复为Java对象 ...
- java序列化测试
0.前言 本文主要对几种常见Java序列化方式进行实现.包括Java原生以流的方法进行的序列化.Json序列化.FastJson序列化.Protobuff序列化. 1.Java原生序列化 Java原生 ...
- 浅析若干Java序列化工具【转】
在Java中socket传输数据时,数据类型往往比较难选择.可能要考虑带宽.跨语言.版本的兼容等问题.比较常见的做法有: 采用java对象的序列化和反序列化 把对象包装成JSON字符串传输 Googl ...
- Java序列化的几种方式
本文着重解说一下Java序列化的相关内容. 假设对Java序列化感兴趣的同学能够研究一下. 一.Java序列化的作用 有的时候我们想要把一个Java对象变成字节流的形式传出去,有的时候我们想要从 ...
- 【转】几种Java序列化方式的实现
0.前言 本文主要对几种常见Java序列化方式进行实现.包括Java原生以流的方法进行的序列化.Json序列化.FastJson序列化.Protobuff序列化. 1.Java原生序列化 Java原生 ...
- (记录)Jedis存放对象和读取对象--Java序列化与反序列化
一.理论分析 在学习Redis中的Jedis这一部分的时候,要使用到Protostuff(Protobuf的Java客户端)这一序列化工具.一开始看到序列化这些字眼的时候,感觉到一头雾水.于是,参考了 ...
随机推荐
- 问题:PHP扩展功能,去掉分号';'没有用,是怎么回事?(已解决)
1. 环境:win10的操作系统,IIS的服务器. 2. 问题描述:PHP要开启访问MYSQL的模块mysqli,我打开配置文件,去掉相关扩展模块前面的分号';',然后重启服务器,但是无效 ~~ 3. ...
- iClock数据服务器默认账号密码
网络空间资产搜索: shodan 搜索 默认密码 admin/a***n End!!!
- ORA-01427: 单行子查询返回多个行 出现原因及对应防止措施
原因:没有做好子查询约束条件 解决方法:1.加 AND ROWNUM =1 条件 但筛选结果可能并非一定是你想要的 2.匹配查询唯一约束条件
- Day14-封装、继承、多态
封装.继承.多态 一.封装 package Demo; //类 private私有 public class student { //属性私有 //名字 private String name; // ...
- SQL Server datetime类型为null的有趣实验
@data1 --变量 测试用 @data2 --当前时间 当@data1为null 则格式转换错误 直接控制台什么也不显示 也不报错 当定义'' 显示默认时间
- Java实现简单的大顶堆
Java实现简单的大顶堆 今天刷LeetCode的347. 前 K 个高频元素的时候使用到了优先队列,由于对大顶堆掌握不算熟练,于是写了一个简单大顶堆练手: 实现代码在最后 之前很少使用泛型来写代码, ...
- 高并发解决方案之 redis 分布式锁
背景:秒杀服务中要写一个定时任务:活动到期时给order微服务发送关闭订单的通知.这需要改变数据库表中的数据,而集群中服务是多节点的方式进行部署,会出现并发执行的情况,所以采用的redis的分布式锁的 ...
- qtconsole和jupyter-notebook的对应启动命令行配置
jupyter-notebook: set conda_path=D:\use\program\Miniconda3 pushd %conda_path% call Scripts\activate. ...
- api进阶Day1文件的创建、删除、访问、设置过滤器并查询。目录的删除、创建。
文件的创建: package file; import java.io.File; import java.io.IOException; /* create:创建 new:新 file:文件 使用F ...
- 基于airtest验证Android端app是否安装及自动化安装
1.检测app是否安装: 使用check_app方法检测是否安装:为什么需要在封装一层做断言呢?主要check_app方法安装成功会返回True,但是未检测到安装时直接报异常了,停止执行.无法直接 ...