单例与序列化的那些事儿
来源:互联网 发布:古筝模拟软件 编辑:程序博客网 时间:2024/06/11 21:40
本文将通过实例+阅读Java源码的方式介绍序列化是如何破坏单例模式的,以及如何避免序列化对单例的破坏。
单例模式,是设计模式中最简单的一种。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。关于单例模式的使用方式,可以阅读单例模式的七种写法
但是,单例模式真的能够实现实例的唯一性吗?
答案是否定的,很多人都知道使用反射可以破坏单例模式,除了反射以外,使用序列化与反序列化也同样会破坏单例。
序列化对单例的破坏
首先来写一个单例的类:
import java.io.Serializable;
/**
* 静态内部类实现单例
*
*/
publicclass Singleton implementsSerializable{
privateSingleton (){}
publicstatic Singleton getSingleton() {
returnSingletonFactory.instance;
}
privatestaticclass SingletonFactory{
private static Singleton instance = new Singleton();
}
}
接下来是一个测试类:
import java.io.*;
public class SerializableDemo1 {
//为了便于理解,忽略关闭流操作及删除文件操作。真正编码时千万不要忘记
//Exception直接抛出
publicstatic void main(String[] args) throwsIOException, ClassNotFoundException {
//Write Obj to file
ObjectOutputStream oos = newObjectOutputStream(newFileOutputStream(“tempFile”));
oos.writeObject(Singleton.getSingleton());
//Read Obj from file
File file = newFile(“tempFile”);
ObjectInputStream ois = newObjectInputStream(newFileInputStream(file));
Singleton newInstance = (Singleton) ois.readObject();
//判断是否是同一个对象
System.out.println(newInstance == Singleton.getSingleton());
}
}
//false
输出结构为false,说明:
通过对Singleton的序列化与反序列化得到的对象是一个新的对象,这就破坏了Singleton的单例性。
这里,在介绍如何解决这个问题之前,我们先来深入分析一下,为什么会这样?在反序列化的过程中到底发生了什么。
ObjectInputStream
对象的序列化过程通过ObjectOutputStream和ObjectInputputStream来实现的,那么带着刚刚的问题,分析一下ObjectInputputStream 的readObject 方法执行情况到底是怎样的。
为了节省篇幅,这里给出ObjectInputStream的readObject的调用栈:
readObject—>readObject0—>readOrdinaryObject—>checkResolve
这里看一下重点代码,readOrdinaryObject方法的代码片段:
privateObject readOrdinaryObject(booleanunshared)
throwsIOException
{
//此处省略部分代码
Object obj; try{ obj = desc.isInstantiable() ? desc.newInstance() : null; }catch(Exception ex) { throw(IOException) newInvalidClassException( desc.forClass().getName(), "unable to create instance").initCause(ex); } //此处省略部分代码 if(obj != null&& handles.lookupException(passHandle) == null&& desc.hasReadResolveMethod()) { Object rep = desc.invokeReadResolve(obj); if(unshared && rep.getClass().isArray()) { rep = cloneArray(rep); } if(rep != obj) { handles.setObject(passHandle, obj = rep); } } returnobj;}
code 3 中主要贴出两部分代码。先分析第一部分:
Object obj;
try{
obj = desc.isInstantiable() ? desc.newInstance() : null;
}catch(Exception ex) {
throw(IOException) newInvalidClassException(desc.forClass().getName(),”unable to create instance”).initCause(ex);
}
这里创建的这个obj对象,就是本方法要返回的对象,也可以暂时理解为是ObjectInputStream的readObject返回的对象。
isInstantiable:如果一个serializable/externalizable的类可以在运行时被实例化,那么该方法就返回true。针对serializable和externalizable我会在其他文章中介绍。
desc.newInstance:该方法通过反射的方式调用无参构造方法新建一个对象。
所以。到目前为止,也就可以解释,为什么序列化可以破坏单例了?
答:序列化会通过反射调用无参数的构造方法创建一个新的对象。
那么,接下来我们再看刚开始留下的问题,如何防止序列化/反序列化破坏单例模式。
防止序列化破坏单例模式
先给出解决方案,然后再具体分析原理:
只要在Singleton类中定义readResolve就可以解决该问题:
import java.io.Serializable;
/**
* Created by hollis on 16/2/5.
* 使用静态内部方式实现单例
*
*/
publicclass Singleton implementsSerializable{
privateSingleton (){}
publicstatic Singleton getSingleton() {
returnSingletonFactory.instance;
}
privatestaticclass SingletonFactory{
private static Singleton instance = new Singleton();
}
privateObject readResolve() {
returnsingleton;
}
}
具体原理,我们回过头继续分析code 3中的第二段代码:
if(obj != null&&
handles.lookupException(passHandle) == null&&
desc.hasReadResolveMethod())
{
Object rep = desc.invokeReadResolve(obj);
if(unshared && rep.getClass().isArray()) {
rep = cloneArray(rep);
}
if(rep != obj) {
handles.setObject(passHandle, obj = rep);
}
}
hasReadResolveMethod:如果实现了serializable 或者 externalizable接口的类中包含readResolve则返回true
invokeReadResolve:通过反射的方式调用要被反序列化的类的readResolve方法。
所以,原理也就清楚了,主要在Singleton中定义readResolve方法,并在该方法中指定要返回的对象的生成策略,就可以方式单例被破坏。
总结
在涉及到序列化的场景时,要格外注意他对单例的破坏。
- 单例与序列化的那些事儿
- 单例与序列化的那些事儿
- 单例与序列化的那些事儿
- 单例与序列化的那些事儿
- 单例与序列化的那些事儿
- 单例与序列化的那些事儿
- 设计模式的那些事儿-----单例模式
- 单例和序列化那些事
- 与内存有关的那些事儿
- UIWebView与JavaScript的那些事儿
- Oracle与DB2的那些事儿
- Javascript与iframe的那些事儿
- UIWebView与JavaScript的那些事儿
- UIWebView与JavaScript的那些事儿
- UIWebView与JavaScript的那些事儿
- UIWebView与JavaScript的那些事儿
- 【与硬盘分区有关的那些事儿】
- XSS与字符编码的那些事儿
- Centos7 安装 RabbitMQ
- 第一行代码-10.2 使用HTTP协议访问网络
- LeetCode-102,103,107,111总结
- 2015年终总结
- 关于索引
- 单例与序列化的那些事儿
- UNIX环境高级编程第十章 信号总结
- 单数组哈希表unordered_map和unordered_set(转)
- tmux设置和使用
- Gstreamer 工具使用 (一)
- 常见的设计模式
- 安卓通知的使用系列7:对话框通知的使用之日期对话框、时间对话框
- spark sql结果写mysql以及隐式转化的简单使用
- TC SRM 683 Div2(1000pts)