[Java]单例模式

简介: 本文介绍了单例模式的概念及其实现方式,包括饿汉式和懒汉式两种形式,并详细探讨了懒汉式中可能出现的线程安全问题及其解决方案,如锁方法、锁代码块和双重检查锁(DCL)。文章通过示例代码帮助读者更好地理解和应用单例模式。

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)
[https://developer.aliyun.com/article/1631322)
出自【进步*于辰的博客

参考笔记一,P28.3、P29.9、P71.1。

1、介绍

1.1 什么是单例模式?

“单例模式”指关闭对外实例化方法,需通过调用类方法获取实例,且多次调用都始终保持同一个实例的一种设计模式。

当一个线程改变此唯一实例的成员变量时,由于其他线程不可见,就会导致并发性问题。因此,往往不声明成员变量,仅定义了成员方法时使用单例模式。

1.2 如何实现单例模式?

看下述代码。(注:此示例未实现单例模式,仅用于说明实现单例模式的思想)

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

获取实例的方法不是通过new或者反射,而是通过调用newInstance()实现。说明:

  1. instance定义为静态私有,一是为了newInstance()可访问,二是防止类外直接获取。
  2. 构造方法声明为private,使得无法主动实例化(new).。
  3. newInstance()是静态公共方法,使用static修饰是因为 Singleton 类无法实例化,故无法通过对象调用newInstance();使用public修饰是为了方便类外调用。

扩展说明:

  1. “懒汉式”是在调用newInstance()时才创建实例,故无需分析。
  2. 判断“饿汉式”的情况:若instance为类变量,其在类初始化时创建,自然可保证唯一实例;若instance为成员变量,其在实例初始化时创建,但由于构造方法禁止实例化,故也是在调用newInstance()时创建,也可保证唯一实例。因此,instance定义为类变量与单例模式没有直接关系。

2、单例模式的两种形式

2.1 饿汉式

基础格式:

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

    public static Singleton newInstance() {
   
        return instance;
    }
}

唯一实例在类内直接创建,不存在多实例可能,不违背“单例”,故不存在线程安全问题,但可能导致内存浪费。

2.2 懒汉式

基础格式:

class Singleton {
   
    private static Singleton instance;
    private Singleton() {
   }

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

newInstance()被调用时,才创建实例。

存在的线程安全问题:
从表面上看,基础“懒汉式”没有问题。可实际上,由于newInstance()本身线程不安全,当多个线程同时调用时,就存在创建多个实例的可能,违背“单例”,故存在线程安全问题。

举个栗子。

public static Singleton newInstance(){
   
    if (instance == null) {
   ---------------A
        instance = new Singleton();-------B
    }
    return instance;
}

假设有两个线程x、y同时调用newInstance()

x先执行A,判断 instance 是否为null,为true,但还未执行B。可此时,x的CPU时间片用完,CPU被y抢去。y也执行A,判断instance是否为null,为true,执行B,创建一个实例。

然后,x重新获得时间片,继续执行,由于x已经判断过instance,故直接执行B,再创建一个实例。

至此,线程x、y都创建了一个实例,这就违背了“单例“。

在高并发下,这种情况很容易发生,而且这只是其中一种情况。

3、解决“懒汉式”线程安全问题的三种方法

3.1 锁方法

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

synchronized同步锁将方法锁住,这样一次就只能有一个线程进入方法。

3.2 锁代码块

public static Singleton newInstance() {
   
    synchronized (Singleton.class) {
   
        if (instance == null) {
   -------------A
            instance = new Singleton();-----B
        }
    }
    return instance;
}

因为可能存在线程安全问题的代码是A,故用同步锁将其锁住,原理与“锁方法”相同。

3.3 DCL

public static Singleton newInstance() {
   
    if (instance == null) {
   --------------------A
        synchronized (Singleton.class) {
   -------B
            if (instance == null) {
   ------------C
                instance = new Singleton();
            }
        }
    }
    return instance;---------------------------D
}

“懒汉式”存在线程安全问题,根本原因就是未对实例存在进行二次判断,这种在两次判断之间介入同步锁(synchronized)进行限制的方法叫做“双重检查锁”(DCL,也称为“双重同步锁”或“双重检测机制”)。

过程推演:
假设有两个线程x、y同时调用newInstance()

x先执行A,判断 instance 是否为null,为true,但还未进入B。可此时,x的CPU时间片用完,CPU被y抢去。y也执行A,判断instance是否为null,为true,进入B,判断instance为null后直接创建一个实例。

然后,x重新获得时间片,待y释放同步锁后进入B,判断instance是否为null,由于y已经创建实例,故instance存在,因此,x释放同步锁执行D返回instance,实例唯一。

注意:必须用同步锁将C锁住,不然与基础“懒汉式”别无二致,无意义。

PS:上面的过程推演只是其中一种情况,作为大家理解DCL的一个推演模板。

最后

本文中的所有例子,是为了阐述“单例模式”思想和如何解决“单例模式”存在的线程安全问题,以及方便大家理解而简单举出的,不一定有实用性,仅是抛砖引玉。

本文完结。

相关文章
|
2月前
|
设计模式 安全 Java
Java编程中的单例模式深入剖析
【10月更文挑战第21天】在Java的世界里,单例模式是设计模式中一个常见而又强大的存在。它确保了一个类只有一个实例,并提供一个全局访问点。本文将深入探讨如何正确实现单例模式,包括常见的实现方式、优缺点分析以及最佳实践,同时也会通过实际代码示例来加深理解。无论你是Java新手还是资深开发者,这篇文章都将为你提供宝贵的见解和技巧。
103 65
|
1月前
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
36 4
|
1月前
|
设计模式 安全 Java
Java编程中的单例模式:理解与实践
【10月更文挑战第31天】在Java的世界里,单例模式是一种优雅的解决方案,它确保一个类只有一个实例,并提供一个全局访问点。本文将深入探讨单例模式的实现方式、使用场景及其优缺点,同时提供代码示例以加深理解。无论你是Java新手还是有经验的开发者,掌握单例模式都将是你技能库中的宝贵财富。
48 2
|
1月前
|
设计模式 安全 Java
Java编程中的单例模式深入解析
【10月更文挑战第31天】在编程世界中,设计模式就像是建筑中的蓝图,它们定义了解决常见问题的最佳实践。本文将通过浅显易懂的语言带你深入了解Java中广泛应用的单例模式,并展示如何实现它。
|
1月前
|
设计模式 SQL 安全
Java编程中的单例模式深入解析
【10月更文挑战第24天】在软件工程中,单例模式是设计模式的一种,它确保一个类只有一个实例,并提供一个全局访问点。本文将探讨如何在Java中使用单例模式,并分析其优缺点以及适用场景。
17 0
|
3月前
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
45 11
|
2月前
|
设计模式 SQL 安全
【编程进阶知识】Java单例模式深度解析:饿汉式与懒汉式实现技巧
本文深入解析了Java单例模式中的饿汉式和懒汉式实现方法,包括它们的特点、实现代码和适用场景。通过静态常量、枚举类、静态代码块等方式实现饿汉式,通过非线程安全、同步方法、同步代码块、双重检查锁定和静态内部类等方式实现懒汉式。文章还对比了各种实现方式的优缺点,帮助读者在实际项目中做出更好的设计决策。
62 0
|
4月前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
3月前
|
设计模式 Java 安全
Java设计模式-单例模式(2)
Java设计模式-单例模式(2)
|
4月前
|
设计模式 安全 Java
Java 单例模式,背后有着何种不为人知的秘密?开启探索之旅,寻找答案!
【8月更文挑战第30天】单例模式确保一个类只有一个实例并提供全局访问点,适用于需全局共享的宝贵资源如数据库连接池、日志记录器等。Java中有多种单例模式实现,包括饿汉式、懒汉式、同步方法和双重检查锁定。饿汉式在类加载时创建实例,懒汉式则在首次调用时创建,后者在多线程环境下需使用同步机制保证线程安全。单例模式有助于提高代码的可维护性和扩展性,应根据需求选择合适实现方式。
37 1