【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)

简介: 设计模式是一种在软件开发中常用的解决复杂问题的方法论。它提供了一套经过验证的解决方案,用于解决特定类型问题的设计和实现。设计模式可以帮助开发人员提高代码的可重用性、可维护性和可扩展性。

1. 前言

设计模式是一种在软件开发中常用的解决复杂问题的方法论。它提供了一套经过验证的解决方案,用于解决特定类型问题的设计和实现。设计模式可以帮助开发人员提高代码的可重用性、可维护性和可扩展性。


设计模式有很多,本文主要介绍单例模式.

08c287700ab546cab2b269f6a08961fe.gif


2. 单例模式

单例模式是一种创建型设计模式,它保证一个类只有一个实例,并提供一个全局访问点来获取该实例。


3. 如何保证一个类只有一个实例

在Java中,通常使用static关键字来保证 类只有唯一的实例.


在单例模式中,类的构造函数被私有化,以防止外部代码直接创建实例。然后,通过一个静态方法或静态变量来获取类的唯一实例。


如果实例不存在,则创建一个新的实例并返回;如果实例已存在,则直接返回该实例。


而是实现上述的方法有很多,下面介绍三种常见的实现模式:


1.饿汉式

2.懒汉式

3.静态内部类



4. 饿汉式单例模式

饿汉式单例: 在类加载时就创建实例,保证在任何情况下都有一个实例可用。


代码示例:

public class Singleton {
    private static final Singleton instance = new Singleton();
    private Singleton() {
    // 私有化构造函数
    }
    public static Singleton getInstance() {
        return instance;
    }
}

代码分析:


df8189cff6b048b0a8837859589fd47c.png



私有化构造函数使得外部无法直接创建实例,如果其它线程想到得到Singleton类的实例,只能通过Singleton类提供的getInstance()方法得到.


5. 懒汉式单例模式

懒汉式单例: 延迟加载实例,只有在需要时才创建实例。


代码示例:


public class SingletonLazy {
    private static SingletonLazy instance = null;
    private SingletonLazy(){
    // 私有化构造方法
    }
    public static SingletonLazy getInstance(){
        if (instance == null){
            instance = new SingletonLazy();
        }
        return instance;
    }
}

代码分析:


82a4eca4944c46e6a3a8c7a1b5fc16e7.png



如果后续代码中没有调用getInstance方法,那么就把创建实例那一步给省下来了.


上述代码中,那种实现单例模式的方式是线程安全的?


答案是 饿汉模式


饿汉模式并没有涉及到修改,而懒汉模式即涉及到读有涉及到修改.那么在多线程模式中就不安全了.


设想一个场景, 如果在懒汉模式下,两个线程同时去调用getInstance方法,如果一个线程读到的instance为null,并给instance进行实例的创建,而另外一个线程读到的instance还是为null,又给instance创建了一次实例.那么instance就被创建多次了


6. 实现线程安全的懒汉式单例

既然懒汉模式在多线程的环境下不安全,那么如果保证懒汉模式在多线程的环境下安全呢?


既然涉及到多线程,就离不开"锁",也就是 synchronized


示例:


bd7a32c556e945fa8b328247d311e9d5.png


通过加锁的方式,解决了线程安全问题,但是又带来了新的问题.


懒汉式单例只是第一次调用getInstance方法时才会发生线程不安全问题.一旦创建好了,线程就安全了.


但上述通过加锁的方式,会导致线程已经安全时仍然时需要加锁,有些多此一举了. 且加锁开销比较大,影响效率.


那么如果保证线程安全时不加锁呢?


实例没有创建前,线程不安全.实例创建之后,线程安全. 那么就可以在加锁操作前在进行一次判断.

public class SingletonLazy {
    private static SingletonLazy instance = null;
    private SingletonLazy(){
    }
    public static SingletonLazy getInstance(){
        if (instance == null) {
            synchronized (SingletonLazy.class) {
                if (instance == null){
                    instance = new SingletonLazy();
                }
            }
        }
        return instance;
    }
}

代码分析:



7a0a29b05a804f7bb2d1cd75e94c0d4d.png


使用双重if保证只创建一次实例以及保证在线程安全时不进行加锁. 这里要重点进行理解.


但仔细思考上述代码仍然有一个问题.


假如两个线程同时调用getInstance方法,第一个线程拿到锁,进入第二层if,开始进行new对象.


这里的new操作可以分为:


1.申请内存,得到内存首地址

2.调用构造方法,来初始化实例

3.把内存的首地址赋值给instance引用

上述的new操作,编译器可能会进行"指令重排序". 就可能会导致无序写入的问题


上述步骤中的2和3,在单线程环境下是可以调换顺序的,并不会影响结果.


但在多线程环境中就会导致无序写入问题:


1.线程A执行到instance = new Singleton();这行代码时,由于指令重排序,可能会先执行分配内存空间、初始化实例对象、将instance指向内存空间这三个操作的任意组合,而不是按照代码顺序执行。

2.假设线程A执行完了分配内存空间和初始化实例对象的操作,但还没有将instance指向内存空间,此时线程B执行到第一个instance == null的判断,结果为false,就会直接返回instance,但此时instance并没有正确初始化。

为了解决上述问题,就可以使用volatile关键字. 它可以禁止指令重排序优化,保证instance的写操作先于读操作,从而避免其他线程在没有正确初始化实例时获取到instance。


public class SingletonLazy {
    private static volatile SingletonLazy instance = null;
    private SingletonLazy(){
    }
    public static SingletonLazy getInstance(){
        if (instance == null) {
            synchronized (SingletonLazy.class) {
                if (instance == null){
                    instance = new SingletonLazy();
                }
            }
        }
        return instance;
    }
}

上述就是一个在多线程环境下完全安全的懒汉式单例模式的写法.


7. 静态内部类实现单例模式

除了上述方式,使用静态内部类也能实现单例模式.


该方式利用静态内部类的特性来实现懒加载和线程安全。


代码示例:

public class Singleton {
    private Singleton() {
        // 私有化构造函数
    }
    private static class SingletonHolder {
        private static final Singleton instance = new Singleton();
    }
    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }
}

静态内部类SingletonHolder持有Singleton的唯一实例,当第一次调用getInstance方法时,才会触发SingletonHolder的初始化,从而创建实例。


由于静态内部类的初始化是线程安全的,因此可以确保在多线程环境下只有一个实例被创建。


8. 总结

本文主要介绍了饿汉式,懒汉式和静态内部类三种实现单例模式的方式,其中 懒汉式单例 很重要,要着重理解双重if的含义.


需要注意的是,以上方式都可以在多线程环境下保证单例的正确创建,但在特殊情况下,如使用反射或序列化/反序列化等机制,仍然可能破坏单例的唯一性。


在实际应用中,需要根据具体场景和需求选择合适的单例模式实现方式。


5cf20061bb5f4d108508ee06f60b0eaf.gif

相关文章
|
2月前
|
设计模式 运维 监控
并发设计模式实战系列(4):线程池
需要建立持续的性能剖析(Profiling)和调优机制。通过以上十二个维度的系统化扩展,构建了一个从。设置合理队列容量/拒绝策略。动态扩容/优化任务处理速度。检查线程栈定位热点代码。调整最大用户进程数限制。CPU占用率100%
178 0
|
2月前
|
算法 Java 调度
Java多线程基础
本文主要讲解多线程相关知识,分为两部分。第一部分涵盖多线程概念(并发与并行、进程与线程)、Java程序运行原理(JVM启动多线程特性)、实现多线程的两种方式(继承Thread类与实现Runnable接口)及其区别。第二部分涉及线程同步(同步锁的应用场景与代码示例)及线程间通信(wait()与notify()方法的使用)。通过多个Demo代码实例,深入浅出地解析多线程的核心知识点,帮助读者掌握其实现与应用技巧。
|
2月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
4月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
169 23
|
3月前
|
数据采集 存储 网络协议
Java HttpClient 多线程爬虫优化方案
Java HttpClient 多线程爬虫优化方案
|
5月前
|
安全 Java 开发者
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
|
7月前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
150 1
Java—多线程实现生产消费者
|
6月前
|
缓存 安全 算法
Java 多线程 面试题
Java 多线程 相关基础面试题
|
7月前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
165 7
|
7月前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。
152 7

热门文章

最新文章