再说单例模式的线程安全问题

Posted CoderBuff

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了再说单例模式的线程安全问题相关的知识,希望对你有一定的参考价值。

今天和同事聊起了单例模式的线程安全,我说如果不做任何措施,单例模式在多线程下是不安全的,得到的“单例”实际上并不是单例。但是为什么不是单例呢?由此我上网查了一下,在使用单例模式时,一定要注意线程安全问题,之前的写法没有任何问题。如下:

 1 package day_5_singleton;
 2 
 3 /**
 4  * 单例
 5  * 
 6  * @author turbo
 7  *
 8  *         2016年9月8日
 9  */
10 public class Singleton {
11     private static Singleton instance;
12 
13     private Singleton() {
14     }
15 
16     public static synchronized Singleton GetInstance() {
17 
18         if (instance == null) {
19             instance = new Singleton();
20         }
21         
22         return instance;
23     }
24 }

问题就在于,synchronized对整个方法加锁,形成同步机制,这样虽然解决了单例模式的线程安全问题,但是却产生另外一个问题性能问题,对方法加锁这个颗粒度有点大,我们稍微改进一下。如下:

 1 package day_5_singleton;
 2 
 3 /**
 4  * 单例
 5  * 
 6  * @author turbo
 7  *
 8  *         2016年9月12日
 9  */
10 public class Singleton {
11     private static Singleton instance;
12 
13     private Singleton() {
14     }
15 
16     public static Singleton GetInstance() {
17 
18         if (instance == null) {
19             synchronized (Singleton.class) {
20                 if (instance == null){
21                     instance = new Singleton();
22                 }
23             }
24         }
25         
26         return instance;
27     }
28 }

利用双重锁的方式这样颗粒度变小了,但还是利用同步的方式来解决资源共享问题。其实这上面两种写法称之为“懒加载”,即在用到的时候再来实例化。

我们再次修改代码,如下。

 1 package day_5_singleton;
 2 
 3 /**
 4  * 单例
 5  * 
 6  * @author turbo
 7  *
 8  *         2016年9月12日
 9  */
10 public class Singleton {
11     private static Singleton instance = new Singleton();
12 
13     private Singleton() {
14     }
15 
16     public static Singleton GetInstance() {
17         return instance;
18     }
19 }

我们不利用线程同步的方式,而是在类被加载的时候就生成一个实例对象。这称之为“勤加载”,这个带来的问题就是,不管这个单例有没有用到都会一直存在。

两者都有其优缺点,但相对于利用线程同步的方式来解决线程安全问题,“勤加载”会是一个较为明智的选择。

以上是关于再说单例模式的线程安全问题的主要内容,如果未能解决你的问题,请参考以下文章

Java设计模式--单例模式

线程安全的单例模式是否真的安全

怎么实现一个线程安全的单例模式

LINUX多线程(线程池,单例模式,线程安全,读者写者模型)

懒汉单例模式出现的线程安全问题(C++)

线程安全的 C# 单例模式