单例模式—四种写法详解
单例定义:
一个类只有一个实例,并提供一个全局访问点。
巧妙利用了编程语言的一些语法规则:构造函数private, 然后提供一个public的方法返回类的一个实例;又方法和返回的类的实例都是static类型,所以只能被类所拥有,而不能被实例化类的对象拥有。这样一个类就只能有一个实例了。
- 最简单的写法(非线程安全,有叫它“懒汉式”的)
public class Singleton { private static Singleton uniqueInstance; private Singleton() {} public static Singleton getInstatnce() { if (uniqueInstance == null) { uniqueInstance = new Singleton(); } return uniqueInstance; } }
2. 加”synchronized“保证多线程下的线程安全(同步代码块,高频访问时,性能较差)
public class Singleton { private static Singleton uniqueInstance; private Singleton() {} public static synchronized Singleton getInstatnce() { if (uniqueInstance == null) { uniqueInstance = new Singleton(); } return uniqueInstance; } }
3. ”急切“或”饿汉“式(线程安全,因为JVM加载此类时立即创建此类的唯一实例)
public class Singleton { private static Singleton uniqueInstance = new Singleton(); private Singleton() {} public static Singleton getInstatnce() { return uniqueInstance; } }
4. ”双重检查加锁“ (面试常问)
public class Singleton { private volatile static Singleton uniqueInstance; private Singleton() {} public static Singleton getInstatnce() { if (uniqueInstance == null){ synchronized (Singleton.class) { if (uniqueInstance == null) { uniqueInstance = new Singleton(); } } } return uniqueInstance; } }
注:第一次检查,如果实例不存在,则进入同步块;进入同步块后再次检查,防止第二个线程在第一个线程执行第二次检查之前,已经创建了一个实例。volatile 关键字将禁止指令重排序。