Java 中的序列化与反序列化

4 篇文章 0 订阅

在Java中, 序列化是一种把java对象状态转化成字节码流形式的机制, 这种机制被应用在RPC, Hibernate, RMI, JPA, EJB 和 JMS等场景中; 序列化的逆向操作是反序列化, 顾名思义, 就是把java对象的字节码流还原到jvm中的运行时对象;


常见的序列化方式

Standard Java Serialization API

Java 平台标准序列化与反序列化, 需要class实现Serializable接口, 再由相关的对象流读写工具[ObjectOutputStream, ObjectInputStream]来执行默认操作实现;
  • java中, 如果想要对一个对象进行标准的序列化与反序列化操作, 那这个对象所属 class 必须实现 Serializable 接口;
  • Serializable 是jdk中一个标识性接口, 既没有成员变量也没有方法, 类似的接口还有 Cloneable, Remote;
  • 父类实现 Serializable 接口, 所有子类都继承实现此接口, 也就同样可以进行序列化与反序列化操作;
  • 序列化和反序列化一个 class 的对象时, 需要有一个 serialVersionUID, 如果我们编码时不在 class 定义里面指定, 序列化时会默认生成,
    这个版本号可以用来标识同一个 class 不同版本, 如果反序列化与序列化时的 serialVersionUID 不一致会报 InvalidClassException 异常;
  • 序列化一个对象时, 如果它包含了没有实现 Serializable 接口的引用类型属性, 那就会在序列化时抛出 NotSerializableException 异常;
  • 序列化的信息格式大致为: class 描述开头标志位|class name| class serialVersionUID|序列化标志位|包含属性成员个数|[{属性类型|属性值}]|结束标识符
  • 如果 class 里面的属性被声明为 transient , 那对应的属性就不会被序列化;
  • 与 Serializable 关联的还有个 Externalizable 接口, 它里面有两个方法, 可以实现序列化与反序列化过程中的自定义控制, 因为 Serializable 本身默认的
    序列化逻辑太重了, 它可能会生成过多的我们不需要内容, 网络传输传递的多了也会更耗时, 所以我们可以自定义序列化哪些需要的属性;
具体实现代码:
  • Master: 实现 Serializable 接口的class, 构造出来的对象可以被进行默认的序列化与反序列化操作;
  • Book: 实现 Externalizable 接口的class, 构造出来的对象可以按照自定义的方式进行序列化与反序列化操作;
  • Serialize: 序列化操作示例;
  • Deserialize: 反序列化的操作示例;

Json(JavaScript Object Notation)

Json是一种轻量级的数据交换格式。 易于人阅读和编写, 也易于机器解析和生成。Json字符串的构建主要基于两种数据结构,对象和数组; 不同的开发语言都实现了各自不同对象Json序列化功能, 在Java中比较好用的 fastjson, gson, jackson等;

Json规范的特性:
  • 只有两种结构:对象内的键值对集合结构和数组,对象用{}表示、内部是”key”:”value”,数组用[]表示,不同值用逗号分开
  • 基本数值有7个: false / null / true / object / array / number / string
  • 基于纯文本,所以对于人类阅读是很友好的。
使用fastjson实现Json操作示例:
  • 被序列化的对象-class无需实现Serializable接口;
  • 依靠class中的构造方法或者get/set方法实现序列化与反序列化操作;
  • 在反序列化过程中, 如果没有无参构造方法, 会调用有参构造方法为对象中的属性赋值;
fastjson目前支持的特性:
  1. QuoteFieldNames: 属性用引号括起来
  2. UseSingleQuotes: 使用单引号
  3. WriteMapNullValue: 输出Map的null值
  4. WriteEnumUsingToString: 枚举属性输出toString的结果
  5. WriteEnumUsingName: 枚举数据输出name
  6. UseISO8601DateFormat: 使用日期格式
  7. WriteNullListAsEmpty: List为空则输出[]
  8. WriteNullStringAsEmpty: String为空则输出””
  9. WriteNullNumberAsZero: Number类型为空则输出0
  10. WriteNullBooleanAsFalse: Boolean类型为空则输出false
  11. SkipTransientField: 忽略transient字段
  12. SortField: 字段排序
  13. WriteTabAsSpecial:
  14. PrettyFormat:
  15. WriteClassName:
  16. DisableCircularReferenceDetect:
  17. WriteSlashAsSpecial:
  18. BrowserCompatible:
  19. WriteDateUseDateFormat:
  20. NotWriteRootClassName:
  21. DisableCheckSpecialChar:
  22. BeanToArray:
  23. WriteNonStringKeyAsString:
  24. NotWriteDefaultValue:
  25. BrowserSecure:
  26. IgnoreNonFieldGetter:
  27. WriteNonStringValueAsString:
  28. IgnoreErrorGetter:
  29. WriteBigDecimalAsPlain:
  30. MapSortField:
具体实现代码:

Protocol buffers

Protocol buffers是一种灵活高效的数据结构化存储格式,用于结构型数据序列化,适合数据存储或RPC数据交互。具有语言无关、平台无关、可扩展的序列化结构数据格式。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java序列化是将对象转换为字节流的过程,以便将其存储在磁盘上或通过网络传输。而反序列化则是将字节流转换回对象的过程。 下面是一个示例,演示如何序列化反序列化一个Java对象: ```java import java.io.*; public class SerializationExample { public static void main(String[] args) { // 创建一个Person对象 Person person = new Person("Alice", 30); try { // 将Person对象序列化到文件 FileOutputStream fileOut = new FileOutputStream("person.ser"); ObjectOutputStream out = new ObjectOutputStream(fileOut); out.writeObject(person); out.close(); fileOut.close(); System.out.println("Person对象已经序列化到person.ser文件"); } catch (IOException e) { e.printStackTrace(); } try { // 将Person对象从文件反序列化出来 FileInputStream fileIn = new FileInputStream("person.ser"); ObjectInputStream in = new ObjectInputStream(fileIn); Person serializedPerson = (Person) in.readObject(); in.close(); fileIn.close(); System.out.println("从person.ser文件反序列化出的Person对象:"); System.out.println("姓名:" + serializedPerson.getName()); System.out.println("年龄:" + serializedPerson.getAge()); } catch (IOException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } } } class Person implements Serializable { private static final long serialVersionUID = 1L; private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } ``` 运行该程序,它会将一个Person对象序列化到文件person.ser,并从该文件反序列化出Person对象,并将其打印出来。 需要注意的是,要将一个对象序列化,该对象的类必须实现java.io.Serializable接口。否则,将会抛出NotSerializableException异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值