JAVA总结---序列化的三种方式
序列化和反序列化
序列化:可以将对象转化成一个字节序列,便于存储。
反序列化:将序列化的字节序列还原
优点:可以实现对象的"持久性”, 所谓持久性就是指对象的生命周期不取决于程序。
序列化需要:
所需类:ObjectInputStream和ObjectOutputStream
方法: readObject()和writeObject();
序列化方式一: 实现Serializable接口(隐式序列化)
通过实现Serializable接口,这种是隐式序列化(不需要手动),这种是最简单的序列化方式,会自动序列化所有非static和 transient关键字修饰的成员变量。
-
class Student implements Serializable{
-
private String name;
-
private int age;
-
public static int QQ = 1234;
-
private transient String address = "CHINA";
-
-
Student(String name, int age ){
-
this.name = name;
-
this.age = age;
-
}
-
public String toString() {
-
return "name: " + name + "\n"
-
+"age: " + age + "\n"
-
+"QQ: " + QQ + "\n"
-
+ "address: " + address;
-
-
}
-
public void SetAge(int age) {
-
this.age = age;
-
}
-
}
-
public class SerializableDemo {
-
public static void main(String[] args) throws IOException, ClassNotFoundException {
-
//创建可序列化对象
-
System.out.println("原来的对象:");
-
Student stu = new Student("Ming", 16);
-
System.out.println(stu);
-
//创建序列化输出流
-
ByteArrayOutputStream buff = new ByteArrayOutputStream();
-
ObjectOutputStream out = new ObjectOutputStream(buff);
-
//将序列化对象存入缓冲区
-
out.writeObject(stu);
-
//修改相关值
-
Student.QQ = 6666; // 发现打印结果QQ的值被改变
-
stu.SetAge(18); //发现值没有被改变
-
//从缓冲区取回被序列化的对象
-
ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(buff.toByteArray()));
-
Student newStu = (Student) in.readObject();
-
System.out.println("序列化后取出的对象:");
-
System.out.println(newStu);
-
-
}
-
}
打印结果:
原来的对象:
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()方法,而且只能通过手动进行序列化,并且两个方法是自动调用的,因此,这个序列化过程是可控的,可以自己选择哪些部分序列化
-
public class Blip implements Externalizable{
-
private int i ;
-
private String s;
-
public Blip() {}
-
public Blip(String x, int a) {
-
System.out.println("Blip (String x, int a)");
-
s = x;
-
i = a;
-
}
-
public String toString() {
-
return s+i;
-
}
-
@Override
-
public void writeExternal(ObjectOutput out) throws IOException {
-
// TODO Auto-generated method stub
-
System.out.println("Blip.writeExternal");
-
out.writeObject(s);
-
out.writeInt(i);
-
//
-
}
-
@Override
-
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
-
// TODO Auto-generated method stub
-
System.out.println("Blip.readExternal");
-
s = (String)in.readObject();
-
i = in.readInt();
-
}
-
public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {
-
System.out.println("Constructing objects");
-
Blip b = new Blip("A Stirng", 47);
-
System.out.println(b);
-
ObjectOutputStream o = new ObjectOutputStream(new FileOutputStream("F://Demo//file1.txt"));
-
System.out.println("保存对象");
-
o.writeObject(b);
-
o.close();
-
//获得对象
-
System.out.println("获取对象");
-
ObjectInputStream in = new ObjectInputStream(new FileInputStream("F://Demo//file1.txt"));
-
System.out.println("Recovering b");
-
b = (Blip)in.readObject();
-
System.out.println(b);
-
}
-
-
}
打印结果为:
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()将获得的值赋给相应的值 --->显式序列化
-
public class SerDemo implements Serializable{
-
public transient int age = 23;
-
public String name ;
-
public SerDemo(){
-
System.out.println("默认构造器。。。");
-
}
-
public SerDemo(String name) {
-
this.name = name;
-
}
-
private void writeObject(ObjectOutputStream stream) throws IOException {
-
stream.defaultWriteObject();
-
stream.writeInt(age);
-
}
-
private void readObject(ObjectInputStream stream) throws ClassNotFoundException, IOException {
-
stream.defaultReadObject();
-
age = stream.readInt();
-
}
-
-
public String toString() {
-
return "年龄" + age + " " + name;
-
}
-
public static void main(String[] args) throws IOException, ClassNotFoundException {
-
SerDemo stu = new SerDemo("Ming");
-
ByteArrayOutputStream bout = new ByteArrayOutputStream();
-
ObjectOutputStream out = new ObjectOutputStream(bout);
-
out.writeObject(stu);
-
ObjectInputStream in = new ObjectInputStream(new ByteArrayInputStream(bout.toByteArray()));
-
SerDemo stu1 = (SerDemo) in.readObject();
-
System.out.println(stu1);
-
}
-
}
打印结果为:
年龄23 Ming
注释掉stream.writeInt(age)和age= stream.readInt()后:
年龄0 Ming
方式三结合了显式和隐式序列化,Ming被正常序列化,由于age被trancient修饰,所以需要显式序列化。
原文地址:https://blog.csdn.net/AHuqihua/article/details/81331316
JAVA总结---序列化的三种方式的更多相关文章
- java 获取时间戳的三种方式
java 获取时间戳的三种方式 CreationTime--2018年7月13日16点29分 Author:Marydon 1.实现方式 方式一:推荐使用 System.currentTimeMi ...
- 如何实现有返回值的多线程 JAVA多线程实现的三种方式
可返回值的任务必须实现Callable接口,类似的,无返回值的任务必须Runnable接口.执行Callable任务后,可以获取一个Future的对象,在该对象上调用get就可以获取到Callable ...
- Java实现线程的三种方式和区别
Java实现线程的三种方式和区别 Java实现线程的三种方式: 继承Thread 实现Runnable接口 实现Callable接口 区别: 第一种方式继承Thread就不能继承其他类了,后面两种可以 ...
- [OpenSource]浅谈.Net和Java互相调用的三种方式
在很多的大型系统开发中,开发工具往往不限制于同一种开发语言,而是会使用多种开发语言的混合型开发.目前Java和.Net都声称自己占85%的市场份额,不管谁对谁错,Java和.Net是目前应用开发的两个 ...
- 浅谈.Net和Java互相调用的三种方式
在很多的大型系统开发中,开发工具往往不限制于同一种开发语言,而是会使用多种开发语言的混合型开发.目前Java和.Net都声称自己占85%的市场份 额,不管谁对谁错,Java和.Net是目前应用开发的两 ...
- Java并发编程:Java创建线程的三种方式
目录 引言 创建线程的三种方式 一.继承Thread类 二.实现Runnable接口 三.使用Callable和Future创建线程 三种方式的对比 引言 在日常开发工作中,多线程开发可以说是必备技能 ...
- java 实现websocket的三种方式
Java中实现websocket常见有以下三种方式: 使用tomcat的websocket实现,需要tomcat 7.x,JEE7的支持. 使用spring的websocket,spring与webs ...
- java创建线程的三种方式及其对比
第一种方法:继承Thread类,重写run()方法,run()方法代表线程要执行的任务.第二种方法:实现Runnable接口,重写run()方法,run()方法代表线程要执行的任务.第三种方法:实现c ...
- AJPFX总结java创建线程的三种方式及其对比
Java中创建线程主要有三种方式: 一.继承Thread类创建线程类 (1)定义Thread类的子类,并重写该类的run方法,该run方法的方法体就代表了线程要完成的任务.因此把run()方法称为执行 ...
随机推荐
- DirectX11笔记(七)--Direct3D渲染3--INDICES AND INDEX BUFFERS
原文:DirectX11笔记(七)--Direct3D渲染3--INDICES AND INDEX BUFFERS 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.cs ...
- 微信端的user-Agent
在iPhone下,返回 Mozilla/5.0 (iPhone; CPU iPhone OS 5_1 like Mac OS X) AppleWebKit/534.46 (KHTML, like Ge ...
- 如何实现已发布app的自动更新
要实现app的自动更新,做两件事情就可以搞定 1.获取当前手机中的app版本号 我们可以通过查询mainbundle中的获取CFBundleVersion NSDictionary *infoDict ...
- SQLServer数据库(二)
数据库设计:就是将数据库中的数据库实体及这些数据库实体之间的关系,进行规划和结构化的过程. 项目开发过程: 需求分析 概要设计 详细设计 代码编写 运行测试 打包发行 数据库的系统分析基本步骤:收集信 ...
- Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 全书总结
原文:Introduction to 3D Game Programming with DirectX 12 学习笔记之 --- 全书总结 本系列文章中可能有很多翻译有问题或者错误的地方:并且有些章节 ...
- iOS从零开始 Code Review
http://www.cocoachina.com/ios/20151117/14208.html 这篇帖子不是通篇介绍Code Review的方法论, 而是前大段记录了我们团队怎么从没有这个习惯到每 ...
- oracle怎么捕获表上的DML语句(不包括select)语句)
可以采用dml触发器,如 CREATE OR REPLACE TRIGGER tr_capt_sql BEFORE DELETE OR INSERT OR UPDATE ON manager.test ...
- Pyhton 单行、多行注释方法
一.python单行注释的符号 井号#常被用作单行注释符号,在代码中使用#时,它右边的任何数据都会被忽略,当做是注释.类似c++的// 二.批量.多行注释的符号 多行注释是用三引号: ”’ 注释内容 ...
- 2019-7-25-VisualStudio-2019-新创建项目添加-git-仓库
title author date CreateTime categories VisualStudio 2019 新创建项目添加 git 仓库 lindexi 2019-7-25 15:8:15 + ...
- @loj - 2091@ 「ZJOI2016」小星星
目录 @description@ @solution@ @accepted code@ @details@ @description@ 小 Y 是一个心灵手巧的女孩子,她喜欢手工制作一些小饰品.她有 ...