将对象序列化和反序列化

对象在java中是以堆的方式存储。有时候需要复制对象或者存储对象,而不是对象的引用,这时候就需要用的对象的序列化和反序列化。

1.序列化

Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。

很详细的博客Java中的序列化Serialable高级详解

简单的来说大概有几点注意事项:

  1. 对象要实现了Serializable 接口
  2. 如果序列化和反序列化的serialVersionUID不同则反序列化失败,因为java是通过这个来进行序列化验证的。因此最好还是要定义serialVersionUID 
  3. 序列化保存的是对象的状态,静态变量属于类的状态,因此 序列化并不保存静态变量
  4. Transient 关键字的作用是控制变量的序列化,在变量声明前加上该关键字,可以阻止该变量被序列化到文件中,在被反序列化后,transient 变量的值被设为初始值,如 int 型的是 0,对象型的是 null
  5. 一个子类实现了 Serializable 接口,它的父类都没有实现 Serializable 接口,序列化该子类对象,然后反序列化后输出父类定义的某变量的数值,该变量数值与序列化时的数值不同。反序列化时,为了构造父对象,只能调用父类的无参构造函数作为默认的父对象。

2.实例

 1 /**
 2  * Copyright &copy; 2012-2014 <a href="https://github.com/thinkgem/jeesite">JeeSite</a> All rights reserved.
 3  */
 4 package com.thinkgem.jeesite.common.utils;
 5 
 6 import java.io.ByteArrayInputStream;
 7 import java.io.ByteArrayOutputStream;
 8 import java.io.ObjectInputStream;
 9 import java.io.ObjectOutputStream;
10 import java.lang.reflect.Method;
11 
12 import org.apache.commons.lang3.StringUtils;
13 
14 /**
15  * 对象操作工具类, 继承org.apache.commons.lang3.ObjectUtils类
16  * @author ThinkGem
17  * @version 2014-6-29
18  */
19 public class ObjectUtils extends org.apache.commons.lang3.ObjectUtils {
20 
21     /**
22      * 注解到对象复制,只复制能匹配上的方法。
23      * @param annotation
24      * @param object
25      */
26     public static void annotationToObject(Object annotation, Object object){
27         if (annotation != null){
28             Class<?> annotationClass = annotation.getClass();
29             Class<?> objectClass = object.getClass();
30             for (Method m : objectClass.getMethods()){
31                 if (StringUtils.startsWith(m.getName(), "set")){
32                     try {
33                         String s = StringUtils.uncapitalize(StringUtils.substring(m.getName(), 3));
34                         Object obj = annotationClass.getMethod(s).invoke(annotation);
35                         if (obj != null && !"".equals(obj.toString())){
36                             if (object == null){
37                                 object = objectClass.newInstance();
38                             }
39                             m.invoke(object, obj);
40                         }
41                     } catch (Exception e) {
42                         // 忽略所有设置失败方法
43                     }
44                 }
45             }
46         }
47     }
48     
49     /**
50      * 序列化对象
51      * @param object
52      * @return
53      */
54     public static byte[] serialize(Object object) {
55         ObjectOutputStream oos = null;
56         ByteArrayOutputStream baos = null;
57         try {
58             if (object != null){
59                 baos = new ByteArrayOutputStream();
60                 oos = new ObjectOutputStream(baos);
61                 oos.writeObject(object);
62                 return baos.toByteArray();
63             }
64         } catch (Exception e) {
65             e.printStackTrace();
66         }
67         return null;
68     }
69 
70     /**
71      * 反序列化对象
72      * @param bytes
73      * @return
74      */
75     public static Object unserialize(byte[] bytes) {
76         ByteArrayInputStream bais = null;
77         try {
78             if (bytes != null && bytes.length > 0){
79                 bais = new ByteArrayInputStream(bytes);
80                 ObjectInputStream ois = new ObjectInputStream(bais);
81                 return ois.readObject();
82             }
83         } catch (Exception e) {
84             e.printStackTrace();
85         }
86         return null;
87     }
88 }

java.io.ObjectOutputStream:表示对象输出流

它的writeObject(Object obj)方法可以对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。

java.io.ObjectInputStream:表示对象输入流

它的readObject()方法源输入流中读取字节序列,再把它们反序列化成为一个对象,并将其返回。

本示例中,将对象序列化为byte数组,首先创造一个ByteArrayOutputStream字节数组输出流,表示输出。然后使用ObjectOutputStream(字节数组输出流)来构造一个对象输出流,表示将对象输出到字节数组输出流中。最后通过 oos.writeObject(object);将object写入字节数组输出流,再转换为字节数组。反序列则相反。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Linux驱动

24.C++- 抽象类(存虚函数)、接口、多重继承

抽象类和接口 什么是抽象类 用来表示现实世界中的抽象概念 是一种只能定义类型,而不能产生对象的类 只能被子类继承,且抽象类的相关成员函数没有完整的体现,用来被子...

3676
来自专栏Java帮帮-微信公众号-技术文章全总结

【选择题】Java基础测试七

【选择题】Java基础测试七 86.欲构造ArrayList类的一个实例,此类继承了List接口,下列哪个方法是正确的?( B ) A、ArrayList m...

4683
来自专栏https://www.cnblogs.com/L

Spark算子篇 --Spark算子之combineByKey详解

第二个函数:一开始a是初始值,b是分组内的元素值,比如A[1_],因为没有b值所以不能调用combine函数,第二组因为函数内元素值是[2_,3]调用combi...

962
来自专栏Java帮帮-微信公众号-技术文章全总结

05-02总结方法,数组(一维)

/* 练习题(1) 数组遍历:就是依次输出数组中的每一个元素。 注意:数组提供了一个属性length,用于获取数组的长度。 格式:数组名.lengt...

3417
来自专栏一个会写诗的程序员的博客

《Kotlin 极简教程 》第6章 泛型

通常情况的类和函数,我们只需要使用具体的类型即可:要么是基本类型,要么是自定义的类。

843
来自专栏老马说编程

(37) 泛型 (下) - 细节和局限性 / 计算机程序的思维逻辑

查看历史文章,请点击上方链接关注公众号。 35节介绍了泛型的基本概念和原理,上节介绍了泛型中的通配符,本节来介绍泛型中的一些细节和局限性。 这些局限性主要与Ja...

1936
来自专栏GreenLeaves

JavaScript之call()和apply()方法详解

简介:apply()和call()都是属于Function.prototype的一个方法属性,它是JavaScript引擎内在实现的方法,因为属于Functio...

1906
来自专栏于晓飞的专栏

Java 泛型进阶

在 List<String> 中添加 Integer 将不会通过编译,但是List<Sring>与List<Integer>在运行时的确是同一种类型。

1443
来自专栏彭湖湾的编程世界

【算法】实现栈和队列

栈(stack) 栈(stack)是一种后进先出(LIFO)的集合类型, 即后来添加的数据会先被删除 ? 可以将其类比于下面文件的取放操作:新到的文件会被先取走...

3246
来自专栏杨建荣的学习笔记

通过shell来比较oracle和java中的字符串使用(r4笔记第49天)

今天在无意中看到了java字符串的一些东西,发现和oracle比较起来还是有一定的意义的,但是发现知识点准备好了,比较的时候,每一处java的变更都得重编译运行...

3295

扫码关注云+社区