单例模式有很多种写法,不同的写法都是适用不同业务场景。但大大小小都会受到多线程安全,稳定性的影响。今天我来讲讲,如何避免这种情况。
先来复习下什么是单例模式?
单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。
注意:
- 1、单例类只能有一个实例。
- 2、单例类必须自己创建自己的唯一实例。
- 3、单例类必须给所有其他对象提供这一实例。
介绍
意图:保证一个类仅有一个实例,并提供一个访问它的全局访问点。
主要解决:一个全局使用的类频繁地创建与销毁。
何时使用:当您想控制实例数目,节省系统资源的时候。
如何解决:判断系统是否已经有这个单例,如果有则返回,如果没有则创建。
关键代码:构造函数是私有的。
public class Singleton {
private volatile static Singleton singleton;
private Singleton (){}
public static Singleton getSingleton() { if (singleton == null)
{ synchronized (Singleton.class) { if (singleton == null)
{ singleton = new Singleton(); } } }
return singleton; } }
单例构造函数私有化,就是为了不让使用者在程序中直接new操作实例化。但还是可以通过反射破坏单例模式
,通过反射获取该类的所有方法包括构造函数。
所以为了避免该可能性,就可以在private Singleton (){} 方法中 加判断
private Singleton (){
if(singleton !=null){
throw new Exception("XXXX");
}
}
抛出错误直接结束通过反射实例化的进程。
同样可以通过反序列化可以操作类的实例化,可以在该类中加:
private Object readResolve() {
return getSingleton();
}
在反序列化执行过程中会执行到ObjectInputStream#readOrdinaryObject方法,这个方法会判断对象是否包含readResolve方法,如果包含的话会直接调用这个方法获得对象实例。
让它直接走加同步锁的方法,就可以避免产生新的实例了。