7种单例模式

一、什么是单例模式?
单例模式,顾名思义就是 一个类只能有一个实例,并且在整个项目中都能访问到这个实例。

二、Java中单例模式的7种写法及其分析
写法一

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

 



最简单的写法,缺点在于实例在类初始化的时候就创建了,如果在整个项目中都没有使用到该类,就会创建内存空间的浪费。

写法二

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

 



解决了写法一在类初始化的时候就创建实例的问题,然而只能在单线程中使用,在多线程中使用如果多个线程同时进入if语句中,就可能出现创建多个实例的问题。

写法三

public class Singleton {

private Singleton(){}

private static Singleton instance;

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

 



解决了写法二可能出现的问题,可以在多线程中使用。缺点在于synchronized关键字会强制一次只能让一个线程进入方法中,其他线程不得不阻塞等待该线程退出方法。

写法四

public class Singleton {

private Singleton(){}

private static Singleton instance;

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

 



解决了写法四出现的问题,缺点在于 JVM 在执行对引用赋值时并不是一个原子操作。具体是首先在堆中为 Singleton 分配内存空间,然后对 Singleton 执行初始化操作, 最后再将引用变量指向该实例所对应的内存地址以完成赋值。而 JVM 对这一步进行了优化,使得步骤二和步骤三可以不按顺序执行。所以就有可能出现一个线程在执行了步骤一之后执行了步骤三,然后另一个线程调用了该方法,此时 instance 引用变量不为空,然而 Singleton 实例还没有完成初始化,就会造成报错。

写法五

public class Singleton {

private Singleton(){}

private static volatile Singleton instance;

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

 



volatile关键字能够禁止指令重排,保证在写操作没有完成之前不能调用读操作。

写法六

public class Singleton {

private Singleton(){}

private static class SingletonHolder {
private static Singleton instance = new Singleton();
}

public static Singleton getInstance(){
return SingletonHolder.instance;
}
}

 



这种写法的好处是充分利用了静态内部类的特点,它的初始化操作跟外部类是分开的。在没有调用 getInstance() 方法之前,静态内部类不会进行初始化,在第一次调用该方法后就生成了唯一一个实例。

写法七

public enum Singleton {

INSTANCE;

public void fun(){}

}

 


推荐写法,简单高效。充分利用枚举类的特性,只定义了一个实例,且枚举类是天然支持多线程的。

posted @ 2019-12-20 11:26  源l  阅读(528)  评论(0编辑  收藏  举报