单例模式

1.什么是单例模式?

  普天之下,为我独尊,一个国家只能有一个皇帝,一个类只产生一个对象。

2.那么单例模式该如何实现呢?

  构造函数私有→其他类不能通过new关键字创建对象
  final关键字→保证对象引用的唯一性
  static关键字→由类本身可以创建对象

3.单例模式的创建(七种)

1)懒汉模式(线程不安全)

public class Singleton{
   private static Singleton instance;
   private Singleton(){}
   public static Singleton getInstance(){
      if(instance == null){
         instance = new Singleton();
      }
      return instance;
   } 
}

2)懒汉模式(加锁)

public class Singleton{
   private static Singleton instance;
   private Singleton(){}
   public static synchronized Singleton getInstance(){
      if(instance == null){
         instance = new Singleton();
      }
      return instance;
   } 
}

3)饿汉模式(线程安全)

public class Singleton{
   private static Singleton instance = new Singleton();
   private Singleton(){}
   public static Singleton getInstance(){
      return instance;
   } 
}

4)饿汉变种

public class Singleton{
   private static Singleton instance = null;
   static{
      instance = new Singleton();
   }
   private Singleton(){}
   public static Singleton getInstance(){
      return this.instance;
   } 
}

和第三种差不多
5)静态内部类

public class Singleton{
   private static class SingletonHolder{
      private static final Singleton INSTANCE = new Singleton();
   }
   private Singleton(){}
   public static final Singleton getInstance(){
      return SingletonHolder.INSTANCE;
   } 
}

  这种方式利用了ClassLoder的机制来保证初始化实例时只有一个线程,它和第三种和第四种不同的是(细微的差别):第三种和第四种只要Singleton类被装载了,那么instance就会被实例化,而这种方式是Singleton被装载了,但instance不一定被初始化。因为SingletonHolder类没有被主动使用,只有显式通过调用getInstance方法时,才会装载SingletonHolder类,从而实例化。
6)枚举类

public enum Singleton{
   INSTANCE;
   public void whateverMethod(){
   }
}

  他不仅能避免多线程问题,而且还能防止反序列化重新创建新的对象。
  
7)双重校验锁

public class Singleton{
   private volatile static Singleton singleton;
   private Singleton(){}
   public static Singleton getSingleton(){
      if(singleton == null){
         synchronized(Singleton.class){
            if(singleton == null){
               singleton = new Singleton();
            }
         }
      }
      return singleton;
   }
}

  双重校验锁机制是说:并不是每次进入getInstance方法都需要同步,而是先不同步,进入方法过后,先检查实例是否存在,如果不存在才进入下面的同步块,这是第一层检查。进入同步块后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例 ,这是第二层检查。这样就只需要同步一次,从而减少了多次在同步情况下进行判断所浪费的时间。

4.关于单例模式的几个问题

1)饿汉模式先出现,为什么后来又有了懒汉模式?
  答:饿汉模式是在类加载时就初始化了,而懒汉模式是在需要的时候才会被初始化。
2)懒汉模式因为不是原子性操作,所以线程不安全。
3)双重校验锁中volatile关键字的作用?
  答:可见性、可屏蔽性(禁止指令重排序)
  a.可见性:当一条线程修改了这个变量的值,新值对其他线程是立即可知的。但并不是在并发下就是安全的,volatile在各个线程的工作内存中不存在一致性问题,但java运算并非原子操作,所以在并发下一样时不安全的。例如:i++,编译为字节码指令后,不一定是一条字节码,底层还有汇编等,所以不是原子性操作。
    原理:volatile修饰的变量赋值后,会多执行一个“lock addl $0x0,(%esp)”操作,这条指令是把EJP容器的值为0,显然这是一个空操作,lock前缀的作用是使本CPU的Cache写入了内存,相当于对Cache中的变量做了一次"store和write"操作,将此变量同步到主内存中
  b.可屏蔽性:也就是内存屏障。
    例子:

int a1=1;
int a2=2;
volatile int a3=3;
int a4=4;

当没有volatile时,第四条语句可能会先于第三条语句执行,但加上volatile,在第三条语句还未执行完毕,第四条语句不会执行。
4)枚举类的问题
  枚举类单例的保证:枚举的构造方法是私有的,我们在访问枚举类实例时会执行构造方法,同时每一个枚举类实例都是static final类型的,也就表明只能被实例化一次。
  实现不被反序列化:在序列化的时候,java仅仅是将枚举对象的name属性输出到结果中,反序列化的时候则是通过java.lang.Eunm的valueOf方法来根据名字查找枚举对象。同时编译器是不允许任何对这种序列化进制的定制的,因此禁用了writeObject、readObject等方法。

你可能感兴趣的:(单例模式)