学习记录-GOF的23种模式_单例模式
单例模式
单例模式是指在程序中,一个类只有一个实例,且该类可自行创建这个实例的一种模式,常见的单例模式如window的任务管理器,无论点击多少次都只会创建一个实例。
单例模式的特点:
1、一个类只有一个实例
2、该单例对象只能有该类自行创建
3、该单例对外提供一个全局访问点
UML类图
一、懒汉式单例模式
懒汉式单例模式指的是在类加载并没有该类实例,当第一次调用getInstance时才会创建实例。
public class LazySingleton {
private static LazySingleton instance = null;
//private避免外部创建实例
private LazySingleton(){
}
//创建实例
public static LazySingleton getInstance(){
if(instance == null){
instance = new LazySingleton();
}
return instance;
}
}
注意:懒汉式单例模式在多线程下使用时存在线程安全问题。
二、双重锁检查单例模式
上述的懒汉式单例模式是存在线程安全问题,为了保证线程安全,则需要对实例加锁,使用volatile和synchronized关键字对实例加锁检查,使用加锁机制后,每次访问都会同步,影响性能。
public class LazySingleton {
/*volatile禁止指令重排序,防止在多线程下,还未初始化便返回
*/
private static volatile LazySingleton instance = null;
//private避免外部创建实例
private LazySingleton(){
}
//在创建实例方法上加上sybchronied
public static LazySingleton getInstance(){
//检查是否为null
if(instance == null ){
synchronized (LazySingleton.class){
//加锁后再次检查
if(instance == null){
instance = new LazySingleton();
}
}
}
return instance;
}
}
第一次检查实例是否为空,是为了提高运行效率,如果实例已经存在这直接返回实例。
synchronized加锁后再次检查是为了防止二次创建实例,例如,线程1和线程2同时进入,当线程1通过了第一次检查时,线程2抢占了线程,此时实例还没创建,线程2也通过了检查创建了实例,如果没有第二次检查,则线程1和线程2皆创建了实例,不符合单例条件了,因此需要做第二次检查。
volatile关键字则是为了防止指令重排序。
三、饿汉式单例模式
饿汉式单例模式指在类加载时便创建了一个实例,后续访问中都只是将实例返回即可。
与懒汉式不同,饿汉式单例模式是线程安全的,它在类加载的时候便创建一个实例,后续便不会在改变,因此线程安全。
public class HungrySingleton {
private static HungrySingleton instance= new HungrySingleton();
private HungrySingleton(){}
public static HungrySingleton getInstance(){
return instance;
}
}
参考地址:http://c.biancheng.net/view/1338.html