序列化和反序列化

序列化:可以将对象转化成一个字节序列,便于存储。

反序列化:将序列化的字节序列还原

优点:可以实现对象的"持久性”, 所谓持久性就是指对象的生命周期不取决于程序。

序列化需要:

所需类:ObjectInputStream和ObjectOutputStream

方法: readObject()和writeObject();

序列化方式一: 实现Serializable接口(隐式序列化)

通过实现Serializable接口,这种是隐式序列化(不需要手动),这种是最简单的序列化方式,会自动序列化所有非static和 transient关键字修饰的成员变量。


  1. class Student implements Serializable{
  2. private String name;
  3. private int age;
  4. public static int QQ = 1234;
  5. private transient String address = "CHINA";
  6. Student(String name, int age ){
  7. this.name = name;
  8. this.age = age;
  9. }
  10. public String toString() {
  11. return "name: " + name + "\n"
  12. +"age: " + age + "\n"
  13. +"QQ: " + QQ + "\n"
  14. + "address: " + address;
  15. }
  16. public void SetAge(int age) {
  17. this.age = age;
  18. }
  19. }
  20. public class SerializableDemo {
  21. public static void main(String[] args) throws IOException, ClassNotFoundException {
  22. //创建可序列化对象
  23. System.out.println("原来的对象:");
  24. Student stu = new Student("Ming", 16);
  25. System.out.println(stu);
  26. //创建序列化输出流
  27. ByteArrayOutputStream buff = new ByteArrayOutputStream();
  28. ObjectOutputStream out = new ObjectOutputStream(buff);
  29. //将序列化对象存入缓冲区
  30. out.writeObject(stu);
  31. //修改相关值
  32. Student.QQ = 6666; // 发现打印结果QQ的值被改变
  33. stu.SetAge(18); //发现值没有被改变
  34. //从缓冲区取回被序列化的对象
  35. ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(buff.toByteArray()));
  36. Student newStu = (Student) in.readObject();
  37. System.out.println("序列化后取出的对象:");
  38. System.out.println(newStu);
  39. }
  40. }

打印结果:
原来的对象:

name: Ming

age: 16

QQ: 1234

address: CHINA

序列化后取出的对象:

name: Ming

age: 16

QQ: 6666

address: null

发现address(被transient)和QQ(被static)也没有被序列化,中途修改QQ的值是为了以防读者误会QQ被序列化了。因为序列化可以保存对象的状态,但是QQ的值被改变了,说明没有被序列化。static成员不属于对象实例,可能被别的对象修改没办法序列化,序列化是序列对象。对于address被反序列化后由于没有对应的引用,所以为null。而且Serializable不会调用构造方法。

PS:细心的可能发现序列化很诱人,可以保存对象的初始信息,在以后可以回到这个初始状态。

序列化方式二:实现Externalizable接口。(显式序列化)

Externalizable接口继承自Serializable, 我们在实现该接口时,必须实现writeExternal()和readExternal()方法,而且只能通过手动进行序列化,并且两个方法是自动调用的,因此,这个序列化过程是可控的,可以自己选择哪些部分序列化


  1. public class Blip implements Externalizable{
  2. private int i ;
  3. private String s;
  4. public Blip() {}
  5. public Blip(String x, int a) {
  6. System.out.println("Blip (String x, int a)");
  7. s = x;
  8. i = a;
  9. }
  10. public String toString() {
  11. return s+i;
  12. }
  13. @Override
  14. public void writeExternal(ObjectOutput out) throws IOException {
  15. // TODO Auto-generated method stub
  16. System.out.println("Blip.writeExternal");
  17. out.writeObject(s);
  18. out.writeInt(i);
  19. //
  20. }
  21. @Override
  22. public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
  23. // TODO Auto-generated method stub
  24. System.out.println("Blip.readExternal");
  25. s = (String)in.readObject();
  26. i = in.readInt();
  27. }
  28. public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
  29. System.out.println("Constructing objects");
  30. Blip b = new Blip("A Stirng", 47);
  31. System.out.println(b);
  32. ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("F://Demo//file1.txt"));
  33. System.out.println("保存对象");
  34. o.writeObject(b);
  35. o.close();
  36. //获得对象
  37. System.out.println("获取对象");
  38. ObjectInputStream in = new ObjectInputStream(new FileInputStream("F://Demo//file1.txt"));
  39. System.out.println("Recovering b");
  40. b = (Blip)in.readObject();
  41. System.out.println(b);
  42. }
  43. }

打印结果为:
Constructing objects

Blip (String x, int a)

A Stirng47

保存对象

Blip.writeExternal

获取对象

Recovering b

Blip.readExternal

A Stirng47

当注释掉writeExternal和readExternal方法后打印结果为:
Constructing objects

Blip (String x, int a)

A Stirng47

保存对象

Blip.writeExternal

获取对象

Recovering b

Blip.readExternal

null0

说明:Externalizable类会调用public的构造函数先初始化对象,在调用所保存的内容将对象还原。假如构造方法不是public则会出现运行时错误。

序列化方式三:实现Serializable接口+添加writeObject()和readObject()方法。(显+隐序列化)

如果想将方式一和方式二的优点都用到的话,可以采用方式三, 先实现Serializable接口,并且添加writeObject()和readObject()方法。注意这里是添加,不是重写或者覆盖。但是添加的这两个方法必须有相应的格式。

  • 1,方法必须要被private修饰                                ----->才能被调用
  • 2,第一行调用默认的defaultRead/WriteObject(); ----->隐式序列化非static和transient
  • 3,调用read/writeObject()将获得的值赋给相应的值  --->显式序列化

  1. public class SerDemo implements Serializable{
  2. public transient int age = 23;
  3. public String name ;
  4. public SerDemo(){
  5. System.out.println("默认构造器。。。");
  6. }
  7. public SerDemo(String name) {
  8. this.name = name;
  9. }
  10. private void writeObject(ObjectOutputStream stream) throws IOException {
  11. stream.defaultWriteObject();
  12. stream.writeInt(age);
  13. }
  14. private void readObject(ObjectInputStream stream) throws ClassNotFoundException, IOException {
  15. stream.defaultReadObject();
  16. age = stream.readInt();
  17. }
  18. public String toString() {
  19. return "年龄" + age + " " + name;
  20. }
  21. public static void main(String[] args) throws IOException, ClassNotFoundException {
  22. SerDemo stu = new SerDemo("Ming");
  23. ByteArrayOutputStream bout = new ByteArrayOutputStream();
  24. ObjectOutputStream out = new ObjectOutputStream(bout);
  25. out.writeObject(stu);
  26. ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(bout.toByteArray()));
  27. SerDemo stu1 = (SerDemo) in.readObject();
  28. System.out.println(stu1);
  29. }
  30. }

打印结果为:
年龄23  Ming

注释掉stream.writeInt(age)和age= stream.readInt()后:
年龄0  Ming

方式三结合了显式和隐式序列化,Ming被正常序列化,由于age被trancient修饰,所以需要显式序列化。

原文地址:https://blog.csdn.net/AHuqihua/article/details/81331316

JAVA总结---序列化的三种方式的更多相关文章

  1. java 获取时间戳的三种方式

      java 获取时间戳的三种方式 CreationTime--2018年7月13日16点29分 Author:Marydon 1.实现方式 方式一:推荐使用 System.currentTimeMi ...

  2. 如何实现有返回值的多线程 JAVA多线程实现的三种方式

    可返回值的任务必须实现Callable接口,类似的,无返回值的任务必须Runnable接口.执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable ...

  3. Java实现线程的三种方式和区别

    Java实现线程的三种方式和区别 Java实现线程的三种方式: 继承Thread 实现Runnable接口 实现Callable接口 区别: 第一种方式继承Thread就不能继承其他类了,后面两种可以 ...

  4. [OpenSource]浅谈.Net和Java互相调用的三种方式

    在很多的大型系统开发中,开发工具往往不限制于同一种开发语言,而是会使用多种开发语言的混合型开发.目前Java和.Net都声称自己占85%的市场份额,不管谁对谁错,Java和.Net是目前应用开发的两个 ...

  5. 浅谈.Net和Java互相调用的三种方式

    在很多的大型系统开发中,开发工具往往不限制于同一种开发语言,而是会使用多种开发语言的混合型开发.目前Java和.Net都声称自己占85%的市场份 额,不管谁对谁错,Java和.Net是目前应用开发的两 ...

  6. Java并发编程:Java创建线程的三种方式

    目录 引言 创建线程的三种方式 一.继承Thread类 二.实现Runnable接口 三.使用Callable和Future创建线程 三种方式的对比 引言 在日常开发工作中,多线程开发可以说是必备技能 ...

  7. java 实现websocket的三种方式

    Java中实现websocket常见有以下三种方式: 使用tomcat的websocket实现,需要tomcat 7.x,JEE7的支持. 使用spring的websocket,spring与webs ...

  8. java创建线程的三种方式及其对比

    第一种方法:继承Thread类,重写run()方法,run()方法代表线程要执行的任务.第二种方法:实现Runnable接口,重写run()方法,run()方法代表线程要执行的任务.第三种方法:实现c ...

  9. AJPFX总结java创建线程的三种方式及其对比

    Java中创建线程主要有三种方式: 一.继承Thread类创建线程类 (1)定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务.因此把run()方法称为执行 ...

随机推荐

  1. jquery判断页面网址是否有效

    方法一:(jQuery方法: 适用所有浏览器) HTML页面: <!DOCTYPE html><html><head lang="en"> &l ...

  2. CentOS7安装步骤

    CentOS7安装详解   本文基于vmware workstations进行CentOS7安装过程展示,关于vmware workstations安装配置本人这里不再介绍,基本过程相当于window ...

  3. 实现移动端上拉加载和下拉刷新的vue插件(mescroll.js)

    做一个简单的移动端展示项目,后台分页后前端加载,实现上拉加载下一页,找了下,还是用这个mescroll.js插件好一点 1.npm安装 npm install --save mescroll.js / ...

  4. BZOJ 3057圣主的考验题解

    老师居然考这么毒瘤的题目!!!!! 很容易想到dp,f[i][j]表示有i个节点,左子树的最深深度为j的方案数 枚举左子树有多少节点然后转移,复杂度为n^3 T飞~ 我们考虑到有深度为h的树的节点有多 ...

  5. Hashkell 第一篇

    心情极差.................. 无事可做,其实是没心情去做事情,只好又翻起了haskell入门 记下几点,以备查询: 1. 函数名首字符是不可以大写的, 而且名称中可以有单引号,这也是合 ...

  6. 【JZOJ3854】【NOIP2014八校联考第2场第2试9.28】分组(group)

    MEi Bsny所在的精灵社区有n个居民,每个居民有一定的地位和年龄,ri表示第i个人的地位,ai表示第i个人的年龄. 最近社区里要举行活动,要求几个人分成一个小组,小组中必须要有一个队长,要成为队长 ...

  7. PHPCMS快速建站系列之 pc标签where中如何使用变量

    {pc:content action="lists" catid="$catid" where="typeid='$t'" order=&q ...

  8. 2017 校赛 问题 B: CZJ-Superman

    题目描述 “那是只鸟?那是飞机?那是——超人!” 程序员在看完<CZJ-Superman>之后,励志要成为一名“CZJ-Superman”,学会了两个特殊技能ZZZ和JJJ,足以成为一名“ ...

  9. HTTP参考

    HTTP参考 一.HTTP码应码响应码由三位十进制数字组成,它们出现在由HTTP服务器发送的响应的第一行. 响应码分五种类型,由它们的第一位数字表示: 1.1xx:信息,请求收到,继续处理 2.2xx ...

  10. 【LeetCode】90.Subsets II

    Subsets II Given a collection of integers that might contain duplicates, nums, return all possible s ...