谈谈Java对象序列化

前言

创建对象时,只要我们需要,便会一直存在。而若是程序终止,那无论如何它都不会继续存在。但某些情况下,如果对象能够在程序不运行时仍能保存其信息,那将非常有用。这样,在下次运行程序时,该对象将被重建并且拥有的信息与在程序上次运行时它所拥有的信息相同。此时,对象序列化便隆重登场了...

关于对象序列化

当我们在内存中创建可复用的Java对象时,一般情况下会依赖于JVM的生命周期,即会随着JVM的停止而销毁。但Java对象序列化可实现JVM停止运行之后保存(持久化)指定的对象,并可随时读取该对象。

引用《Thinking in Java》里面的一段话:
利用它(对象序列化)可以实现轻量级持久性,“持久性”意味着一个对象的生存周期并不取决于程序是否正在执行;它可生存于程序的调用之间。

解析对象序列化

序列化分为两部分:序列化反序列化

  • 序列化:将数据分解成字节流,以便在文件或网络上传输。(即 Java Object 转成 byte[] )
  • 反序列化:打开字节流并重构对象,从而取出对象。(即 byte[] 转成 Java Object )

一般用途

1. 将内存中的对象状态保存至文件或数据库
2. 网络传输
3. Java远程方法调用(RMI)

具体实例

(一)将对象状态保存至文件或数据库

  1. 实现 java.io.Serializable 接口
        File file = new File("person.out");  
        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(file));    //创建一个写入指定OutputStream的ObjectOutputStream对象
        Person person = new Person("Hello", 101, Gender.MALE);  
        out.writeObject(person);    //将指定的对象写入
        out.close();  
 
        ObjectInputStream in = new ObjectInputStream(new FileInputStream(file));  
        Object newPerson = in.readObject();    //从ObjectInputStream读取对象
        in.close();  
        System.out.println(newPerson); 

分析:
上述是网上常见的一个例子,非常简单。实现
Serializable 接口采用的是默认序列化机制,我们注意到以上例子使用了两个类 ObjectOutputStreamObjectInputStream ,该demo就是使用了 ObjectOutputStream 来持久化对象,而使用 ObjectInputStream 从文件中取出对象。

影响序列化的因素:transient关键字
(1) 当某个字段被声明为transient后,默认序列化机制就会忽略该字段。
(2) 若是想运用其它方法来序列化这个标了transient的字段,则可以通过添加两个方法:writeObject()与readObject()。
  1. 实现 java.io. Externalizable 接口
    JDK中提供了另一个序列化接口——
    Externalizable ,** Externalizable** 继承于 Serializable ,序列化的细节需要程序员完成。使用该接口之后,之前基于 Serializable 接口的序列化机制就将失效。
    @Override 
    public void writeExternal(ObjectOutput out) throws IOException {  
 
    }  
 
    @Override 
    public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {  
 
    }

(二) 使用 Protostuff 序列化库

在开发项目中,序列化是经常需要处理的问题。不管是网络传输上用json、xml、对象序列化,还是缓存数据中的序列化,都是相当重要的。

前提

以并发系统为例,使用 Redis 缓存存储数据时,若单单只使用JDK的序列化,效率会很低,对并发类系统造成的影响无法想象。而Google开发的一套序列化方案 Protostuff ,好处很多,独立于语言,独立于平台,重要的是效率相当高,使用 Protostuff 序列化后的字节大小是json的10分之一,xml的20分之一,是二进制序列化的10分之一。

目的

以实际的并发系统为例,使用 Protostuff 框架进行序列化处理。

说明

  • 开发环境:IDEA,Tomcat
  • 软件环境:SpringMVC,Spring,MyBatis
  • 模块描述:若缓存中有我们需要的对象,则取出来;若是没有,则存入缓存中( Redis )

过程

  • 序列化
set Object() -> 序列化 -> byte[]
  • 反序列化
get -> byte[] -> 反序列化 -> Object()

编码

  1. Maven 引入 Protostuff
<!-- protostuff序列化依赖 -->
        <dependency>
            <groupId>com.dyuproject.protostuff</groupId>
            <artifactId>protostuff-core</artifactId>
            <version>1.0.8</version>
        </dependency>

        <dependency>
            <groupId>com.dyuproject.protostuff</groupId>
            <artifactId>protostuff-runtime</artifactId>
            <version>1.0.8</version>
        </dependency>
  1. 在Dao层缓存RedisDao中引入 Protostuff

set Object() -> 序列化 -> byte[]

        //LinkedBuffer:缓存器
        byte[] bytes = ProtobufIOUtil.toByteArray(seckill, schema,
                LinkedBuffer.allocate(LinkedBuffer.DEFAULT_BUFFER_SIZE));

        //超时缓存
        int timeout = 60 * 60;  //1小时
        String result = jedis.setex(key.getBytes(), timeout, bytes);

get -> byte[] -> 反序列化 -> Object()

        byte[] bytes = jedis.get(key.getBytes());

        //从缓存获取到
        if (bytes != null) {
            //空对象
            Seckill seckill = schema.newMessage();
            ProtobufIOUtil.mergeFrom(bytes, seckill, schema);
                return seckill;
        }

测试

结果表明,字节数被压缩得极小,在并发系统效率确实不错,很值得学习的一个序列化库!

总结

本博文的内容均为自己的总结,想深入学习的朋友可参考《Thinking in Java》和《Effective Java》,里面写的内容非常详细的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,723评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,080评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,604评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,440评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,431评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,499评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,893评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,541评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,751评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,547评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,619评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,320评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,890评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,896评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,137评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,796评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,335评论 2 342

推荐阅读更多精彩内容

  • JAVA序列化机制的深入研究 对象序列化的最主要的用处就是在传递,和保存对象(object)的时候,保证对象的完整...
    时待吾阅读 10,834评论 0 24
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,579评论 18 139
  • (一)Java部分 1、列举出JAVA中6个比较常用的包【天威诚信面试题】 【参考答案】 java.lang;ja...
    独云阅读 7,059评论 0 62
  • 想起以前读过的一个经典故事: 一位老人家门口有一片公共草坪,他非常享受坐在草坪上晒太阳的安静时光。结果有一天,一群...
    踢车刘阅读 463评论 0 1
  • 阁楼远处朝阳起,篱笆墙外草木稀。 昨夜梦中琴瑟鼓,纷扰斯人几多思。
    就把心酸说成笑阅读 242评论 0 1