JAVA实现单例模式有5种方案:分别是“懒汉”“饿汉”“枚举”“内部静态类”“双重校验锁”。由于是单例模式,因此不能将类的构造函数暴露在外面,所以要将构造函数重写为私有的。同时需要考虑多线程安全问题:多个线程不能构造出多个类对象。
懒汉:
public class Singleton{
private static Singleton instance=null;
private Singleton(){}
public synchronized static Singleton getInstance(){
if(instance==null)
instance=new Singleton();
return instance;
}
}
上面的代码中,synchronized就是为了保证多线程安全问题的,但由于粒度有点粗,影响性能。
饿汉:
public class Singleton{
private static Singleton instance=new Singleton();
private Singleton(){}
public static Singleton getInstance(){
return instance;
}
}
对比懒汉模式和饿汉模式,相信大家能看出他们名字的含义了。懒汉是需要时候初始化对象,饿汉是在装载的时候初始化对象。
枚举:(推荐)
public enum Singleton{
INSTANCE;
public static Singleton getINstance(){
return INSTANCE;
}
}
内部静态类:(推荐)
public class Singleton{
private static class SingletonHolder{
private static final Singleton INSTANCE=new Singleton();
}
private Singleton(){}
public static Singleton getInstance(){
return SingletonHolder.INSTANCE;
}
}
在内部静态类方案中,对象的加载也是懒汉模式,使用内部类的classloader机制来保证线程安全性。在java虚拟机中,类的生命周期包括装载连接和初始化。jvm会保证对类的初始化过程被同步的进行。
懒汉模式的另一种写法为:
public class Singleton {
private static Singleton instance=null;
private Singleton(){}
public static Singleton getInstance(){
synchronized (Singleton.class) {
if(instance==null)
instance=new Singleton();
}
return instance;
}
}
在上面的写法中略作改动就是:
双重校验锁(DCL)(正确的写法):
public class Singleton{
private volatile static Singleton instance=null;//一定要加volatile
private Singleton(){}
public static Singleton getInstance(){
if(instance==null){
synchronized (Singleton.class) {
if(instance==null)
instance=new Singleton();
}
}
return instance;
}
}
在JMM相关讨论中,双重校验锁曝光率很高。早期的DCL的确是错误的,但是随着volatile在DCL中的使用,上面的DCL已经是正确的写法了。但是大家都不太使用。关于DCL的来龙去脉,请参加博客http://yizhenn.iteye.com/admin/blogs/2294121
上面就是单例模式中5种常用的实现方式。推荐使用枚举和内部静态类。
另外,单利类在实现序列化时候一定要重写private Object readResolve() throws ObjectStreamException方法!!!
到知乎阅读最新的技术博客:https://www.zhihu.com/people/hulianwangzhaopin/activities
笔者开设了一个知乎live,详细的介绍的JAVA从入门到精通该如何学,学什么?
想深入学习和提高JAVA能力的同学,欢迎收听https://www.zhihu.com/lives/932192204248682496
提供给想学习云计算的同学,欢迎收听https://www.zhihu.com/lives/1046567982750281728