单例模式想必大家都已经很熟悉了,通常它的实现方式分为以下两种:
//懒汉式实现
public class Singleton{
private static Singleton instance = null;
private Singleton(){}
public static newInstance(){
if(null == instance){
instance = new Singleton();
}
return instance;
}
public void doSomething(){
// Do something ...
}
}
//饿汉式
public class Singleton{
private static Singleton instance = new Singleton();
private Singleton(){}
public static Singleton newInstance(){
return instance;
}
public void doSomething(){
// Do something ...
}
}
在单利模式的使用中,懒汉式和饿汉式有不同的应用场景,如果创建单利对象很占内存但是不是在应用启动是必须使用的,我们一般会使用懒汉式,等到真的需要使用单利时才会去创建它,如果随着项目启动要立即使用我们会使用饿汉模式在初始化是创建单利对象。
上述两种模式中,如果在多线程的情况下,饿汉式不会出现问题,因为JVM只会加载一次单利类,但是懒汉式可能就会出现重复创建单利对象的问题,是线程不安全的。
那有没有办法,使饿汉式的单利模式也是线程安全的呢?答案肯定是有的,大家通常会使用加同步锁的方式去实现,但是这样实现起来比较麻烦,我们可以利用JVM的类加载机制去实现。在很多情况下JVM已经为我们提供了同步控制,比如:
a.在static{}区块中初始化的数据
b.访问final字段时 等等。
在JVM进行类加载的时候他会保证数据是同步的,我们可以这样实现:
采用类级内部类,在这个内部类里面去创建对象实例。这样的话,只要不使用类级内部类他就不会去创建对象实例,从而实现懒汉式的延迟加载和线程安全。
public class Singleton{
//内部类,在装载该内部类时才会去创建单利对象
private static class SingletonHolder{
public static Singleton instance = new Singleton();
}
private Singleton(){}
public static Singleton newInstance(){
return SingletonHolder.instance;
}
public void doSomething(){
//do something
}
}
这样就可以实现线程安全的饿汉式单利模式。
另外我们还可以通过枚举类型来实现单利模式,这也是比较推荐的方式。
使用枚举类型实现单例模式如下:
public enum Singleton{
//定义一个枚举的元素,它就是Singleton的一个实例
instance;
public void doSomething(){
// do something ...
}
}
Ok,单例模式先介绍到这里。