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

这篇具有很好参考价值的文章主要介绍了【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

1. 前言

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

设计模式有很多,本文主要介绍单例模式.
【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)

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;
    }
}

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

私有化构造函数使得外部无法直接创建实例,如果其它线程想到得到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;
    }
}

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

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

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

答案是 饿汉模式

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

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

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

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

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

示例:
【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)
通过加锁的方式,解决了线程安全问题,但是又带来了新的问题.

懒汉式单例只是第一次调用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;
    }
}

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

使用双重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的含义.

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

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

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

感谢你的观看!希望这篇文章能帮到你!
专栏: 《从零开始的Java学习之旅》在不断更新中,欢迎订阅!
“愿与君共勉,携手共进!”
【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)文章来源地址https://www.toymoban.com/news/detail-501651.html

到了这里,关于【Java|多线程与高并发】设计模式-单例模式(饿汉式,懒汉式和静态内部类)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【设计模式】详解单例设计模式(包含并发、JVM)

    在软件开发中,经常需要某些类 只能有唯一的实例 ,比如数据库连接。如何才能保证整个应用中只有一个唯一实例?如果靠人为制定的协定来约束,显然不能很好的保证这一点。如果要从 语法上约束 ,在面向对象里面,什么地方能够约束实例的创建? 显然,只有 构造函数

    2024年02月15日
    浏览(36)
  • 设计模式(单例模式,工厂模式),线程池

    目录 什么是设计模式? 单例模式 饿汉模式 懒汉模式 工厂模式 线程池 线程池种类 ThreadPoolExcutor的构造方法: 手动实现一个线程池  计算机行业程序员水平层次不齐,为了 让所有人都能够写出规范的代码, 于是就有了设计模式, 针对一些典型的场景,给出一些典型的解决方案 单例

    2024年02月11日
    浏览(41)
  • 设计模式3:单例模式:静态内部类模式是怎么保证单例且线程安全的?

    上篇文章:设计模式3:单例模式:静态内部类单例模式简单测试了静态内部类单例模式,确实只生成了一个实例。我们继续深入理解。 静态变量什么时候被初始化? 这行代码 private static Manager instance = new Manager(); 什么时候执行? 编译期间将.java文件转为.class文件,运行期间

    2024年02月12日
    浏览(46)
  • 【JAVA开发面试】如何处理并发访问如何进行代码的单元测试Java多线程编程消息中间件设计模式技术难题是如何解决的

    【 点我-这里送书 】 本人详解 作者:王文峰,参加过 CSDN 2020年度博客之星,《Java王大师王天师》 公众号:JAVA开发王大师,专注于天道酬勤的 Java 开发问题 中国国学、传统文化和代码爱好者的程序人生,期待你的关注和支持!本人外号:神秘小峯 山峯 转载说明:务必注明

    2024年02月03日
    浏览(52)
  • 【Linux】简单线程池的设计与实现 -- 单例模式

    线程池: 一种线程使用模式。线程过多会带来调度开销,进而影响缓存局部性和整体性能。而 线程池维护着多个线程,等待着监督管理者分配可并发执行的任务 。这避免了在处理短时间任务时创建与销毁线程的代价。线程池不仅能够保证内核的充分利用,还能防止过分调度。

    2024年02月12日
    浏览(44)
  • 【Linux】线程池设计/单例模式/STL、智能指针与线程安全/读者写者问题

    线程池:一种线程使用模式。线程过多会带来调度开销,进而影响缓存局部性和整体性能。而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务。这避免了在处理短时间任务时创建与销毁线程的代价。线程池不仅能够保证内核的充分利用,还能防止过分调度。可

    2024年02月03日
    浏览(45)
  • 线程安全版本的单例设计模式 与 生产者消费者模型简介

    目录 单例设计模式 单例设计模式——饿汉式 单例设计模式——懒汉式 单例设计模式——懒汉式(优化步骤) 生产者消费者模型 介绍 优点 补充:关于阻塞队列 单例设计模式能够保证 某个类的实例在程序运行过程中始终都只会存在一份 。这一点在很多场景上都有需要,比

    2023年04月24日
    浏览(60)
  • Java设计模式-单例模式

    单例模式是一种设计模式,它确保一个类只能创建一个实例,并提供一种全局访问这个实例的方式。在Java中,单例模式可以通过多种方式来实现,其中最常见的是使用私有构造函数和静态方法实现 在Java中,实现单例模式的方式有多种,其中最常见的实现方式包括以下几种:

    2024年02月01日
    浏览(46)
  • 【java】设计模式——单例模式

    单例模式要点 : 一个类只需要一个实例化对象; 必须自行创建实例; 必须自行向整个系统提供这个实例 实现 : 只提供 私有 构造方法; 有一个该类的 静态 私有对象; 提供一个静态 公有 方法用于创建、获取静态私有对象; 分析: 私有构造方法-不能随意创建实例; 静态

    2024年02月13日
    浏览(47)
  • JAVA设计模式——单例模式

    单例模式是应用最广的设计模式之一,也是程序员最熟悉的一个设计模式,使用单例模式的类必须保证只能有创建一个对象。 今天主要是回顾一下单例模式,主要是想搞懂以下几个问题 为什么要使用单例? 如何实现一个单例? 单例存在哪些问题? 单例对象的作用域的范围

    2024年02月16日
    浏览(44)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包