如何用Java写一个线程安全的单例模式?
在Java中,实现线程安全的单例模式有多种方法。下面是两种常见的方法:
1. 饿汉式(静态初始化)
这种方法非常简单,因为单例的实例被声明为static
和final
变量,第一次加载类到内存中时就会初始化,所以创建实例本身是线程安全的。
java复制代码
public class Singleton { | |
// 在类加载时就完成了初始化,所以类加载较慢,但获取对象的速度快 | |
private static final Singleton instance = new Singleton(); | |
// 构造方法私有化,防止外部通过new Singleton()创建对象 | |
private Singleton() {} | |
// 提供一个全局的静态方法,返回Singleton实例 | |
public static Singleton getInstance() { | |
return instance; | |
} | |
} |
2. 双重检查锁定(Double-Checked Locking)
这种方法首先检查实例是否已经创建,如果已经创建则直接返回,否则再进行同步检查。这种方式既保证了线程安全,又避免了不必要的同步开销。
java复制代码
public class Singleton { | |
// volatile保证多线程正确处理instance变量 | |
private static volatile Singleton instance; | |
private Singleton() {} | |
public static Singleton getInstance() { | |
if (instance == null) { // 第一次检查实例是否存在,如果不存在才进入下面的同步块 | |
synchronized (Singleton.class) { // 同步块,线程安全的创建实例 | |
if (instance == null) { // 第二次检查实例是否存在,如果不存在,才创建实例 | |
instance = new Singleton(); | |
} | |
} | |
} | |
return instance; | |
} | |
} |
注意:在Java 5及之前的版本中,双重检查锁定由于JVM的内存模型原因可能会有问题。但在Java 5之后的版本中,由于引入了volatile
关键字和java.util.concurrent
包,双重检查锁定已经可以正确工作。在这个例子中,我们使用volatile
关键字来确保instance
变量的可见性和禁止指令重排序。
这两种方法都可以实现线程安全的单例模式,你可以根据具体的应用场景和需求来选择适合的方法。