欢迎浏览高耳机的博客
希望我们彼此都有更好的收获
感谢三连支持!
单例模式 Singleton是一种常用的软件模式,确保一个类只有一个实例,并提供一个全局访问方法来获取这个实例。这种模式广泛应用于需要控制实例化次数的场景,如数据库连接池、配置管理、日志记录等。本文我们将重点讨论懒汉模式的实现.
单例模式的常用实现方式
饿汉模式:当你使用搜索引擎搜索词条时,不论你是否上下浏览或者翻页,搜索引擎将加载全部的搜索结果,此时在类加载时就完成了实例化。通常这种模式比较消耗资源和时间.
懒汉模式:当你使用搜索引擎搜索词条时,浏览器只会加载当前屏幕页面所能展示内容的上限.只有当你上下浏览,缩放或者进行翻页操作时,才会加载更多结果.这就是在需要时才创建实例。这样做可以加快加载速度,节省了时间和资源.
饿汉模式单例实现
//饿汉模式实现的基本单例模式
class Singleton {
private static Singleton instance = new Singleton();
public static Singleton getInstance() {
return instance;
}
// 单例模式的最关键部分.
private Singleton() { }
}
private SingletonLazy() { }
这行代码的作用是定义一个私有的构造函数。这是实现单例模式的一个关键步骤,具体作用如下:
限制实例化:通过将构造函数设置为私有(private
),可以防止外部代码通过new
关键字直接创建该类的实例。这样可以确保只有单例类自己能够创建其对象,控制了实例的创建过程。
保证唯一性:单例模式的核心目标是确保一个类只有一个实例。通过私有化构造函数,可以防止外部通过构造函数创建多个实例,保证了实例的唯一性。
控制创建时机:在单例模式中,通常在内部提供一种特殊的方法(如getInstance()
)来控制实例的创建时机。这意味着实例的创建可以被延迟,直到真正需要时才进行,这有助于优化资源的使用。
封装和安全性:私有构造函数还有助于封装类的实现细节,确保类的用户不能改变实例的创建方式,增加了代码的安全性。
在单例模式的实现中,私有构造函数通常与静态方法(如getInstance()
)结合使用,该静态方法负责检查是否已经存在一个实例,如果不存在,则创建一个新实例并返回。这样,无论何时何地调用getInstance()
,都只会得到同一个实例。
懒汉模式单例实现
class SingletonLazy {
// 此处先把这个实例的引用设为 null, 先不着急创建实例.
private static SingletonLazy instance = null;
public static SingletonLazy getInstance() {
if (instance == null) {
instance = new SingletonLazy();
}
return instance;
}
private SingletonLazy() { }
}
在这个懒汉模式单例的基本实现中,其实存在着严重的线程安全问题 !
在多线程环境下,如果两个或更多的线程同时执行getInstance()
方法,并且此时instance
变量为null
,那么每个线程都可能看到instance
是null
,并且每个线程都可能创建一个新的实例。这样就违背了单例模式的原则,即只创建一个实例。为了确保线程安全,我们必须采取措施:
class SingletonLazy {
// 此处先把这个实例的引用设为 null, 先不着急创建实例.
private static volatile SingletonLazy instance = null;
private static Object locker = new Object();
public static SingletonLazy getInstance() {
// 在这个条件中判定当前是否应该要加锁.
if (instance == null) {
synchronized (locker) {
if (instance == null) {
instance = new SingletonLazy();
}
}
}
return instance;
}
private SingletonLazy() { }
}
关键点解释
volatile关键字:确保多线程环境下的可见性和禁止指令重排序。
双重校验锁:第一次检查instance
是否为null
,避免不必要的同步;第二次检查确保只有一个实例被创建。
同步块:确保只有一个线程可以进入创建实例的代码块。
测试单例
验证创建的单例对象是否为同一对象:
public class Demo16 {
public static void main(String[] args) {
SingletonLazy s1 = SingletonLazy.getInstance();
SingletonLazy s2 = SingletonLazy.getInstance();
System.out.println(s1 == s2);
}
}
单例模式总结
控制实例化次数:节约资源,提高性能。
全局访问点:便于管理和访问。
全局状态:可能导致代码难以测试和维护。
线程安全问题:需要额外处理多线程环境下的同步问题。
通过本文的介绍,我们了解了单例模式的理论基础、实现方式、线程安全问题。掌握单例模式不仅能够帮助我们写出更高效、更优雅的代码,还能够提升我们的系统设计能力。在未来的开发工作中,合理运用单例模式,将使我们的设计更加成熟和稳定。
希望这篇博客能为你理解单例模式提供一些帮助。
如有不足之处请多多指出。
我是高耳机。