[Java]单例模式

简介: 本篇文章主要阐述单例模式的基础实现,重心在于如何解决单例模式中的“线程安全”问题。不涉及理论。如果文中阐述不全或不对的,多多交流。

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)

https://developer.aliyun.com/article/1440675
出自【 进步*于辰的博客


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

1、什么是单例模式?

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


注意:

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

2、如何实现单例模式?

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

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

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


  1. instance定义为静态私有,(1)、为了newInstance()可访问;(2)、防止类外直接获取。
    扩展分析“懒汉式”是在调用newInstance()时才创建实例,故不必多言。判断“饿汉式”的情况:若instance为类变量,其在类初始化时创建,自然可保证唯一实例;若instance为成员变量,其在实例初始化时创建,但由于构造方法禁止实例化,故也是在调用newInstance()时创建,也可保证唯一实例。因此,instance定义为类变量与单例模式没有直接关系。
  2. 构造方法声明为private,使无法主动实例化(new).。
  3. newInstance()是静态公共方法,使用static修饰是因为 Singleton 类无法实例化,故无法通过对象调用newInstance();使用public修饰是为了方便类外调用。

3、单例模式的两种形式

3.1 形式一:“饿汉式”

基础格式:

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

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

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

3.2 形式二:“懒汉式”

基础格式:

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

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

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 都创建了一个实例,这就违背了“单例“。


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

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

4.1 方法一:“锁方法”

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

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

4.2 方法二:“锁代码块”

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

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

4.3 方法三:“双重检测机制”

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

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


过程推演:

假设有两个线程 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上面的过程推演只是其中一种情况,作为大家理解双重检测机制的一个推演模板。

5、最后

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

本文完结。

相关文章
|
3月前
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
42 4
|
1月前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
21天前
|
设计模式 安全 Java
Java 单例模式,背后有着何种不为人知的秘密?开启探索之旅,寻找答案!
【8月更文挑战第30天】单例模式确保一个类只有一个实例并提供全局访问点,适用于需全局共享的宝贵资源如数据库连接池、日志记录器等。Java中有多种单例模式实现,包括饿汉式、懒汉式、同步方法和双重检查锁定。饿汉式在类加载时创建实例,懒汉式则在首次调用时创建,后者在多线程环境下需使用同步机制保证线程安全。单例模式有助于提高代码的可维护性和扩展性,应根据需求选择合适实现方式。
29 1
|
24天前
|
SQL 设计模式 安全
Java编程中的单例模式深入解析
【8月更文挑战第27天】本文旨在探索Java中实现单例模式的多种方式,并分析其优缺点。我们将通过代码示例,展示如何在不同的场景下选择最合适的单例模式实现方法,以及如何避免常见的陷阱。
|
20天前
|
设计模式 安全 Java
Java编程中的单例模式深度解析
【8月更文挑战第31天】 单例模式,作为设计模式中的经典之一,在Java编程实践中扮演着重要的角色。本文将通过简洁易懂的语言,逐步引导读者理解单例模式的本质、实现方法及其在实际应用中的重要性。从基础概念出发,到代码示例,再到高级应用,我们将一起探索这一模式如何优雅地解决资源共享和性能优化的问题。
|
20天前
|
设计模式 安全 Java
Java中的单例模式:理解与实践
【8月更文挑战第31天】在软件设计中,单例模式是一种常用的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。本文将深入探讨Java中实现单例模式的不同方法,包括懒汉式、饿汉式、双重校验锁以及静态内部类等方法。每种方法都有其适用场景和潜在问题,我们将通过代码示例来展示如何根据具体需求选择合适的实现方式。
|
20天前
|
设计模式 安全 Java
Java编程中的单例模式实现与应用
【8月更文挑战第31天】在Java的世界里,单例模式是构建高效且资源友好应用的基石之一。本文将深入浅出地介绍如何通过单例模式确保类只有一个实例,并提供一个全局访问点。我们将探索多种实现方法,包括懒汉式、饿汉式和双重校验锁,同时也会讨论单例模式在多线程环境下的表现。无论你是Java新手还是资深开发者,这篇文章都将为你打开一扇理解并有效应用单例模式的大门。
|
29天前
|
设计模式 SQL 缓存
Java编程中的设计模式:单例模式的深入理解与应用
【8月更文挑战第22天】 在Java的世界里,设计模式是构建可维护、可扩展和灵活的软件系统的基石。本文将深入浅出地探讨单例模式这一经典设计模式,揭示其背后的哲学思想,并通过实例演示如何在Java项目中有效运用。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇洞悉软件设计深层逻辑的大门。
26 0
|
2月前
|
设计模式 安全 Java
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
60 1
|
1月前
|
设计模式 SQL 安全
单例模式大全:细说七种线程安全的Java单例实现,及数种打破单例的手段!
设计模式,这是编程中的灵魂,用好不同的设计模式,能使你的代码更优雅/健壮、维护性更强、灵活性更高,而众多设计模式中最出名、最广为人知的就是Singleton Pattern单例模式。通过单例模式,我们就可以避免由于多个实例的创建和销毁带来的额外开销,本文就来一起聊聊单例模式。