关于对Java单例模式的理解与简述

这篇具有很好参考价值的文章主要介绍了关于对Java单例模式的理解与简述。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://www.cnblogs.com/cnb-yuchen/p/17954739 出自【进步*于辰的博客】

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

目录
  • 1、什么是单例模式?
  • 2、如何实现单例模式?
  • 3、单例模式的两种形式
    • 3.1 形式一:“饿汉式”
    • 3.2 形式二:“懒汉式”
  • 4、解决“懒汉式”线程安全问题的三种方法
    • 4.1 方法一:“锁方法”
    • 4.2 方法二:“锁代码块”
    • 4.3 方法三:“双重检测机制”
  • 5、最后

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 锁住,不然与基础“懒汉式”别无二致,无意义。

上面的过程推演只是其中一种情况,作为大家理解双重检测机制的一个推演模板。

5、最后

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

本文完结。文章来源地址https://www.toymoban.com/news/detail-777096.html

到了这里,关于关于对Java单例模式的理解与简述的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

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

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

相关文章

  • 关于对【oracle索引】的理解与简述

    【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://blog.csdn.net/m0_69908381/article/details/131094864 出自【进步*于辰的博客】 无论 oracle 、 mysql ,亦或者其他数据库,几乎所有企业级项目都会使用 索引 ,因为这能大大提升程序性能。 oracle 索引如何实现

    2024年02月08日
    浏览(24)
  • 关于对【mysql存储过程】的理解与简述

    【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://blog.csdn.net/m0_69908381/article/details/130857854 出自【进步*于辰的博客】 存储过程的细节很多,而在实际工作中又未必都能涉及这些细节,工作时间一长,就可能忘记,于是特来写这篇文章,既是为自

    2024年02月07日
    浏览(25)
  • 011:Mapbox GL两种方式隐藏logo和版权,个性化版权的声明

    第011个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+mapbox中用两种方式隐藏logo和版权,并个性化版权的声明 。 直接复制下面的 vue+mapbox源代码,操作2分钟即可运行实现效果 示例效果

    2023年04月17日
    浏览(38)
  • 单例模式--理解

    单例模式是指在内存中只会创建且仅创建一次对象的设计模式。在程序中多次使用同一个对象且作用相同时,为了防止频繁地创建对象使得内存飙升,单例模式可以让程序仅在内存中创建一个对象,让所有需要调用的地方都共享这一单例对象。 单例模式有两种类型: 懒汉式

    2024年04月12日
    浏览(21)
  • 如何理解单例模式?

    单例模式(Singleton Pattern):采取一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法。 通俗点来讲:就是一个男人只能有一个老婆,一个女人只能有一个老公 单例模式一共有8种方式实现,下面一一举例: 实现步

    2023年04月08日
    浏览(25)
  • 关于单例模式

    单例模式的目的: 单例模式的目的和其他的设计模式的目的都是一样的,都是为了降低对象之间的耦合性,增加代码的可复用性,可维护性和可扩展性。 单例模式: 单例模式是一种常用的设计模式,用简单的言语说, 单例模式就是一个类只包含一个实例对象,且该类能够自

    2024年02月10日
    浏览(23)
  • 如何理解单例模式----饿汉式?

    目录 1.前言 2.本质 3.代码默写 在面试中,理解和掌握单例模式是非常重要的。本文旨在帮助读者深入理解饿汉式单例模式,并通过简洁明了的解释和示例代码,使读者能够轻松掌握并默写出饿汉式单例模式的代码实现。 饿汉式单例模式是一种单例设计模式的实现方式,它在

    2024年01月18日
    浏览(30)
  • 深入理解设计模式-创建型之单例模式

    如果有些数据在系统中应该且只能保存一份,那就应该设计为单例类。 如:配置类:在系统中,我们只有一个配置文件,当配置文件被加载到内存之后,应该被映射为一个唯一的【配置实例】,此时就可以使用单例,当然也可以不用。 全局计数器:我们使用一个全局的计数

    2024年02月12日
    浏览(47)
  • Unity单例模式较为简单的理解

    本文将介绍unity中单例模式从最简单到复杂的实际使用。 单例模式是一种设计模式。 设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软

    2023年04月11日
    浏览(26)
  • 【并发专题】单例模式的线程安全(进阶理解篇)

    最近学习了JVM之后,总感觉知识掌握不够深,所以想通过分析经典的【懒汉式单例】来加深一下理解。(主要是【静态内部类】实现单例的方式)。 如果小白想理解单例的话,也能看我这篇文章。我也通过了【前置知识】跟【普通懒汉式】、【双检锁懒汉】、【静态内部类】

    2024年02月14日
    浏览(31)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包