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客户端)这一序列化工具.一开始看到序列化这些字眼的时候,感觉到一头雾水.于是,参考了 ...
随机推荐
- 解决java.sql.SQLException: null, message from server: "Host 'XXX' is not allowed to connect异常
Sqoop连接MySQL报异常.这个异常是数据库只允许localhost或127.0.0.1访问,不允许远程访问.我用的本机IP都不行. 解决办法:修改访问权限即可. 打开cmd,进入mysql fl ...
- PHP Redis - List (列表)
Redis列表是简单的字符串列表,按照插入顺序排序. 一个列表最多可以包含 232-1 个元素 (4294967295, 每个列表超过40亿个元素) 插入元素在列表头部(lPush,Lpushx) ...
- uniapp+vue3+ts
1. 创建vue3的默认uniapp模板 2. npm init 创建package.json
- 【xUtils框架问题】xUtils继承基类的x.view().inject(this)绑定点击事件@Event无效
由于看得教程里的xUtils比较老了,不知道什么版本的. 还是使用ViewUtils.inject()进行绑定反射的,使用@OnClick进行点击监听绑定的. 3.9.0版本的xUtils使用: x. ...
- Jmeter七、jmeter中的参数化
参数化是为了更好的模拟真实的业务场景 CSV data set config组件 1.更容易使用和理解 2.适合大参数量场景 3.设置方便灵活 EOF=end of file 没有找到文件 comma ...
- white album句子
1.不论是真心的笑,还是真心的生气,我都做不到.我只是个胆小的骗子.
- C#.NET系列●接口抽象类
一.接口基本概念 (1)接口:是把公共方法和属性组合起来,以封装特定功能的一个集合.创建接口时,一般一大写的I开头,接口中的成员都是公有的. 接口定义如下: interface IClown //写一 ...
- 如何卸载powermill?怎么把powermill彻底卸载删除干净重新安装的方法【转载】
如何卸载powermill?怎么把powermill彻底卸载删除干净重新安装的方法.powermill显示已安装或者报错出现提示安装未完成某些产品无法安装的问题,怎么完全彻底删除清理干净powermi ...
- Python pexpect 库的简单使用
一.Python pexpect 库的使用 在终端中许多命令都有与用户交互的场景,例如切换用户时需要手动输入密码,安装应用有时要输入默认配置等.这对 shell 自动化脚本十分不便.expect 命令 ...
- Arduino教程目录
目录 第一节.安装Arduino开发环境 第二节.第一个HelloWorld 第二节续.LED操作 呼吸灯 流水灯 正在加快制作,大家可以先看下面的视频了解基本语法,我准备基础课程和实际项目结合讲解. ...