单例模式
单例模式是Java中比较常见的设计模式,它的目的就是确保一个类只有一个实例,自行实例化并向整个系统提供这个实例。
它有一下几个特征:
- 单例类只能有一个实例;
- 单例类必须自己创建自己的唯一实例;
- 单例类必须给所有其他对象提供这一实例;
单例模式分为一下几种:
懒汉式,也是常用的形式。
public class Singleton {
//声明单例类的对象
private static Singleton singleton = null;
//私有化构造函数,避免外界创建该类的对象
private Singleton() {
}
//向外部对象提供单例类对象
public static Singleton getSingleton() {
if (null == singleton) {
//如果单例类的对象为空,则实例化单例类的对象
singleton = new Singleton();
}
return singleton;
}
}
这种懒汉式没有考虑到线程安全问题,在多线程下可能会不能正常工作
public class Singleton {
private static Singleton singleton = null;
private Singleton() {
}
public static synchronized Singleton getSingleton() {
if (null == singleton) {
singleton = new Singleton();
}
return singleton;
}
}
这种写法能够在多线程中很好的工作,而且看起来它也具备很好的lazy loading,但是,遗憾的是,效率很低,99%情况下不需要同步。
饿汉式
public class Singleton {
// 在自己内部定义自己的一个实例,只供内部调用
private static final Singleton singleton = new Singleton();
private Singleton() {
}
// 这里提供了一个供外部访问本class的静态方法,可以直接访问
public static Singleton getInstance() {
return singleton;
}
}
这种方式基于classloder机制避免了多线程的同步问题,不过,singleton在类装载时就实例化,虽然导致类装载的原因有很多种,在单例模式中大多数都是调用getInstance方法, 但是也不能确定有其他的方式(或者其他的静态方法)导致类装载,这时候初始化instance显然没有达到lazy loading的效果。
public class Singleton {
private static Singleton singleton = null;
static {
singleton = new Singleton();
}
private Singleton() {
}
public static Singleton getInstance() {
return singleton;
}
}
表面上看起来差别挺大,其实和上面那种方式差不多,都是在类初始化即实例化singleton。
双重锁的形式。
public class Singleton {
//声明单例类的对象
private static Singleton singleton = null;
//私有化构造函数,避免外界创建该类的对象
private Singleton() {
}
//向外部对象提供单例类的对象,
public static Singleton getSingleton() {
if (null == singleton) {
synchronized (Singleton.class) {
//同步锁后,再进行一次singleton对象的判断
if (null == singleton) {
//如果单例类的对象为空,则实例化单例类的对象
singleton = new Singleton();
}
}
}
return singleton;
}
}