文章目录
- 简介
- 问题
- 1. 确保一个类只有一个实例
- 2. 为该实例提供全局访问点
- 解决方案
- 示例
- 重构前:
- 重构后:
- 拓展
- volatile 在单例模式中的双重作用
- 总结
简介
单例是一种创建型设计模式,它可以确保一个类只有一个实例,同时为该实例提供全局访问点。
问题
单例模式同时解决了两个问题:
1. 确保一个类只有一个实例
最常见的场景是控制对某些共享资源(例如数据库或文件)的访问。假设你已经创建了一个对象,又要创建一个相同类的对象。你不会得到一个新的对象,而是会得到你已经创建的对象。这种行为是无法通过常规构造函数实现,因为构造函数调用在设计上必须始终返回一个新对象。
2. 为该实例提供全局访问点
全局变量非常方便,但很不安全,因为任何代码都可能覆盖这些变量的内容并使程序崩溃。单例模式类似全局变量,允许你从程序中的任何位置访问某个对象。但是,它还可以保护这个实例不被其他代码覆盖。还有一点,为了不让实现问题1 的代码分散在各个地方,要把它限制在一个类中,特别是当你的其余代码已经依赖了它的时候。
解决方案
所有 Singleton 的实现都有这样两个共同的步骤:
- 把默认构造函数设为私有,防止其他对象使用new创建它。
- 创建一个充当构造函数的静态创建方法。这个方法会调用私有构造函数来创建一个对象并把它缓存在静态字段中。这个方法的所有后续调用都会返回缓存好的对象。
如果你的代码能访问 Singleton 类,那么它就可以调用 Singleton 的静态方法。无论何时调用该方法,都会返回相同的对象。
示例
数据库连接
重构前:
class DBUtil {public Connection getConn() {return DriverManager.getConnection(URL); // 每次新建连接消耗500ms+}
}// 调用端
new DBUtil().getConn().execute("SELECT...");
new DBUtil().getConn().execute("UPDATE..."); // 产生两个独立连接
重构后:
public class Database {private static volatile Database instance; private Connection connection;// 私有化构造并建立物理连接private Database() {this.connection = DriverManager.getConnection(JDBC_URL); // 真实连接建立}// 双重检查锁定实现线程安全public static Database getInstance() {if (instance == null) { synchronized (Database.class) {if (instance == null) {instance = new Database(); }}}return instance;}// 统一入口方法(可扩展缓存逻辑)public ResultSet query(String sql) {return connection.createStatement().executeQuery(sql); // 所有SQL通过单连接执行}
}
拓展
volatile 在单例模式中的双重作用
- 可见性保证(Visibility)
阻止线程的本地缓存与主内存数据不同步,确保所有线程读取到的是最新实例状态。 - 禁止指令重排序(Happens-Before)
消除 JVM 级别可能的危险优化(非原子化对象构造的三步指令):
未加 volatile 时的风险时序:
A线程: 分配内存 → 写入未初始化的对象引用(指令排序导致)
B线程: 获取到非空引用 → 访问未完成初始化的对象(空指针异常)volatile 强制时序:
分配内存 → 初始化对象 → 写入引用(三步骤原子性可见)
总结
- 单例(Singleton)类:声明了一个叫做getInstance获的静态方法来返回实例。单例的构造函数必须为私有。调用获取实例方法必须是获取单例对象的唯一方式。