iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >分析Java中的单例模式
  • 203
分享到

分析Java中的单例模式

2023-06-25 12:06:57 203人浏览 独家记忆
摘要

本篇内容主要讲解“分析Java中的单例模式”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“分析Java中的单例模式”吧!WHAT维基百科给出了解释、实现的思路以及应该注意的地方:单例模式,也叫单子

本篇内容主要讲解“分析Java中的单例模式”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“分析Java中的单例模式”吧!

    WHAT

    维基百科给出了解释、实现的思路以及应该注意的地方:

    单例模式,也叫单子模式,是一种常用的软件设计模式,属于创建型模式的一种。在应用这个模式时,单例对象的类必须保证只有一个实例存在。
    实现单例模式的思路是:一个类能返回对象一个引用(永远是同一个)和一个获得该实例的方法(必须是静态方法,通常使用getInstance这个名称);当我们调用这个方法时,如果类持有的引用不为空就返回这个引用,如果类保持的引用为空就创建该类的实例并将实例的引用赋予该类保持的引用;同时我们还将该类的构造函数定义为私有方法,这样其他处的代码就无法通过调用该类的构造函数来实例化该类的对象,只有通过该类提供的静态方法来得到该类的唯一实例。
    单例模式在多线程的应用场合下必须小心使用。如果当唯一实例尚未创建时,有两个线程同时调用创建方法,那么它们同时没有检测到唯一实例的存在,从而同时各自创建了一个实例,这样就有两个实例被构造出来,从而违反了单例模式中实例唯一的原则。解决这个问题的办法是为指示类是否已经实例化的变量提供一个互斥(虽然这样会降低效率)。

    类图是:

    分析Java中的单例模式

    WHY

    正如定义所说,单例模式就是整个内存模型中,只有一个实例。实例少了,内存占用就少。同时,只有一个实例,也就只需要构建一个对象,计算就少。对于构造过程中需要大量计算或者占用大量资源的对象,只创建一次,就减少了资源占用和内存占用。

    饿汉式

    饿汉式是最简单的一种实现,在类装载过程中,完成实例化,避免多线程问题。

    实现一:静态实例参数与静态代码块

    public class EagerSingleton {    private static final EagerSingleton INSTANCE = new EagerSingleton();    private EagerSingleton() {    }    public static EagerSingleton getInstance() {        return INSTANCE;    }}

    根据java的特性,饿汉式还可以变种写法,有的地方称为静态代码块方式:

    public class EagerSingleton {    private static EagerSingleton INSTANCE = null;    static {        INSTANCE = new EagerSingleton();    }    private EagerSingleton() {    }    public static EagerSingleton getInstance() {        return INSTANCE;    }}

    这两种方式只是在写法上的区别,优缺点没有区别,只是借助Java语言特性的不同写法,所以归为一类。

    饿汉式有两个明显的缺点:

    1. 类装载过程即完成实例化,如果整个应用生命周期内,实例没有使用,也就是浪费资源了。

    2. 因为没有办法向构造函数传递不同的参数,如果需要通过个性化参数定制实例时,这种方式就不支持了。

    实现二:静态内部类

    针对饿汉式第一个缺点,我们可以借助静态内部类的方式,将对象实例化的时间延后。

    public class EagerSingleton {    private EagerSingleton() {    }    private static class EagerSingletonInstance {        private static final EagerSingleton INSTANCE = new EagerSingleton();    }    public static EagerSingleton getInstance() {        return EagerSingletonInstance.INSTANCE;    }}

    但是,依然不能很好的解决第二个缺点,如果需要根据不同的参数实现不同的实例,可以采用下面说的懒汉式实现。

    懒汉式

    懒汉式比饿汉式的一个优点,就是能够在使用的时候再进行实例化。但是,馅饼总是要伴随着陷阱,懒汉式写法有更多的坑,一不小心就摔着了。

    错误一:单线程实现

    public class LazySingleton {    private static LazySingleton INSTANCE = null;    private LazySingleton() {    }    public static LazySingleton getInstance() {        if (INSTANCE == null) {            INSTANCE = new LazySingleton();        }        return INSTANCE;    }}

    之所以定义为单线程实现,是因为 INSTANCE==null这个判断,一个线程通过这个判断,开始进行对象实例化,但是还没有实例化完成,另一个线程又来了,这个时候,对象还没有实例化,就也会开始进行实例化,造成不必要的浪费。

    错误二:同步方法

    public class LazySingleton {    private static LazySingleton INSTANCE = null;    private LazySingleton() {    }    public static synchronized LazySingleton getInstance() {        if (INSTANCE == null) {            INSTANCE = new LazySingleton();        }        return INSTANCE;    }}

    这种方式解决了多线程的问题,但是也引入了新的性能问题:太慢。synchronized把整个方法包起来,也就是每个线程进入的时候,都需要等待其他线程结束调用,才能拿到实例,在性能敏感的场景,是比较致命的。

    错误三:同步代码块之单次检查

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

    这种写法看似将同步代码缩小,但也缩小了多线程保障,也犯了第一种写法的错误,属于没有对多线程有基本了解写出的低级错误代码。

    错误四:同步代码块之双重检查

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

    这种写法在一定程度上属于正确的写法,双重判断可以很好的实现线程安全和延迟加载。如果到这里就结束,那就是谬以千里的毫厘之差。

    双重检查和同步代码块都没有问题,问题出在 INSTANCE=newLazySingleton()这句话。在JVM中,为了充分利用CPU计算能力,会进行重排序优化, INSTANCE=newLazySingleton()做了三件事:

    1. 为 INSTANCE 初始化栈空间

    2. 为 LazySingleton 分配内存空间,实例化对象

    3. INSTANCE 指向 LazySingleton 实例分配的内存空间

    因为重排序优化的存在,真正执行的过程中,可能会出现1-2-3的顺序,也可能出现1-3-2的顺序。如果是1-3-2,INSTANCE 指向了 LazySingleton 实例分配的内存空间后,就不是null,另外一个线程进入判断null时,就会直接返回 INSTANCE,但是这个时候 LazySingleton 实例化还没有完成,就可能出现意想不到的异常。

    正确:双重检查+阻止重排序

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

    这种写法比上面那种,就差在 volatile这个关键字。

    枚举

    懒汉式和饿汉式都能够适用于多线程并发场景,但是通过反序列化或反射可以实例化对象,这样依然不能满足单例模式的要求,所以可以借助枚举实现,枚举可以完美避免多线程并发问题,而且可以防止反序列化和反射创建新对象。第一次看到这样定义单例模式,是在《Effective Java》中,多读经典书还是挺好的。

    public enum EnumSingleton {    INSTANCE;    public void method1() {        // do something    }    public Object method2() {        // do something and return something else        return new Object();    }}

    开发实践中,枚举可以满足绝大部分场景,而且写法简单,定义单例的逻辑只需要三行代码,简洁而不简单,三行代码可以保证线程安全。同时枚举的反序列化只是通过name查找对象,不会产生新的对象;根据JVM规范,通过反射创建枚举对象时,会抛出 IllegalArgumentException异常。这样,相当于通过语法糖防止反序列化和反射破坏单例。

    场景

    1. 无状态工具类:这种工具类不需要记录状态,只保证正确的应用就行,可以通过单例模式来定义。

    2. 数据共享:即多个不相关的两个线程或者进程之间实现通信。因为是一个实例,如果它的属性或者变量值被修改,所有引用都是同时修改的,当然需要 volatile 来定义变量。比如网站的计数器。

    3. 日志应用:通常应用会向日志文件写日志信息,为了实时向文件写,通常会使用单例模式,保证有一个实例持有文件,然后进行操作。

    4. 数据库连接池:数据库连接是一种数据库资源,使用数据库连接池,主要是节省打开或者关闭数据库连接所引起的效率损耗,这种效率上的损耗还是非常昂贵的,通过单例模式来维护,就可以大大降低这种损耗。

    5. WEB应用的配置对象:读取文件需要消耗时间,如果读取大文件,消耗的时间和资源更久,所以通过单例模式可以大大降低消耗。

    6. 。。。

    单例模式的场景还是比较多的,这里只是列出里几个简单的应用场景,算是抛砖引玉,如果看官们有什么其他应用场景,可以在说一说

    到此,相信大家对“分析Java中的单例模式”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

    --结束END--

    本文标题: 分析Java中的单例模式

    本文链接: https://www.lsjlt.com/news/304727.html(转载时请注明来源链接)

    有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

    本篇文章演示代码以及资料文档资料下载

    下载Word文档到电脑,方便收藏和打印~

    下载Word文档
    猜你喜欢
    • 分析Java中的单例模式
      本篇内容主要讲解“分析Java中的单例模式”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“分析Java中的单例模式”吧!WHAT维基百科给出了解释、实现的思路以及应该注意的地方:单例模式,也叫单子...
      99+
      2023-06-25
    • Java中的单例模式实例分析
      本篇内容介绍了“Java中的单例模式实例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1、定义单例模式(Singleton Patter...
      99+
      2023-06-29
    • Java单例模式分析
      目录单例模式为什么要用单例单例的关键点几种写法懒汉式饿汉式静态内部类写法枚举单例容器实现单例参考总结单例模式 为什么要用单例 确保某个类只有一个对象,常用于访问数据库操作,服务的配置...
      99+
      2022-11-12
    • java单例模式的示例分析
      这篇文章主要介绍了java单例模式的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。定义:      &nb...
      99+
      2023-05-30
      java
    • JAVA的单例模式实例分析
      这篇文章主要介绍“JAVA的单例模式实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JAVA的单例模式实例分析”文章能帮助大家解决问题。一、单例模式是什么?单例(Singleton)模式的定义...
      99+
      2023-06-29
    • Java设计模式的单例模式实例分析
      本文小编为大家详细介绍“Java设计模式的单例模式实例分析”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java设计模式的单例模式实例分析”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。什么是单例模式单例模式(S...
      99+
      2023-06-29
    • 用实例分析Java单例模式
      这篇文章主要用实例分析Java单例模式,内容简而易懂,希望大家可以学习一下,学习完之后肯定会有收获的,下面让小编带大家一起来看看吧。本文实例讲述了Java单例模式。分享给大家供大家参考,具体如下:在实际开发的时候会有一些需求,在某个类中只能...
      99+
      2023-05-31
      java 单例模式 ava
    • Java中单例模式与多线程的示例分析
      这篇文章主要介绍了Java中单例模式与多线程的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。单例模式与多线程单例模式就是全局唯一但是所有程序都可以使用的对象写单例模式...
      99+
      2023-06-20
    • PHP中单例模式的示例分析
      这篇文章将为大家详细讲解有关PHP中单例模式的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。一、什么是单例模式?含义     作为对象的创建模式,单例模式确保某一个类...
      99+
      2023-06-20
    • C#单例模式的示例分析
      这篇文章给大家分享的是有关C#单例模式的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。单例模式也是创建型模式的一种,也是23种设计模式中比较简单的一种。见名思意,在整个软件系统中,只有某个类型的一个对象,...
      99+
      2023-06-29
    • C++单例模式实例分析
      本篇内容介绍了“C++单例模式实例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!不能被拷贝的类拷贝只会放生在两个场景中:拷贝构造函数以及...
      99+
      2023-06-29
    • Java 实例解析单例模式
      目录单例模式的介绍优点缺点SynchronizedSynchronized示例Synchronized与非SynchronizedSingleton第一个示例第二个示例第三个示例第四...
      99+
      2022-11-12
    • Java设计者模式简单工厂模式的示例分析
      这篇文章主要介绍Java设计者模式简单工厂模式的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!简介简单工厂模式 (Simple Factory) 又叫静态工厂方法(Static Factory Method)...
      99+
      2023-05-30
      java
    • Java中的四种单例模式浅析
      前言近期在做支付,一开始图省事,也是为了调试方便,支付的alipayClient和tradeService都是使用的时候去拿,这样就会导致创建多次。为了节省资源,统一配置成单例模式。什么是单例Singleton(单例)是设计模式的一种,为了...
      99+
      2023-05-31
      java 单例模式 ava
    • web前端中单例模式的示例分析
      小编给大家分享一下web前端中单例模式的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!单例模式(Singleton Pa...
      99+
      2022-10-19
    • 分析java中全面的单例模式多种实现方式
      目录一、单例模式的思想二、单例模式的 N 种实现方式2.1、饿汉式(线程安全,可用)2.2、常量式(线程安全,可用)2.3、懒汉式(线程不安全,并发场景不可用)2.4、同步的懒汉式?...
      99+
      2022-11-12
    • java模板模式的示例分析
      这篇文章主要介绍java模板模式的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Java设计模式-模板模式什么是模板模式?模板模式,顾名思义,就是通过模板拓印的方式。定义模板,就是定义框架、结构、原型。定义一...
      99+
      2023-06-20
    • java设计模式之单例模式解析
      单例模式是最简单但同时也是很重要的一种设计模式,优点有以下几个方面:当内存占用特别大的类需要频繁地创建销毁时,单例模式可以节省内存和提高性能,例如myBatis里面的sessionFactory当需要对文件做单一读写时,例如同一时间只能同时...
      99+
      2023-05-31
      java 设计模式 单例模式
    • JAVA 枚举单例模式及源码分析的实例详解
      JAVA 枚举单例模式及源码分析的实例详解      单例模式的实现有很多种,网上也分析了如今实现单利模式最好用枚举,好处不外乎三点:线程安全不会因为序列化而产生新实例防止反射攻击但是貌似没...
      99+
      2023-05-31
      java 枚举 单例
    • java设计模式中策略模式的示例分析
      小编给大家分享一下java设计模式中策略模式的示例分析,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!策略模式策略模式(Strategy Pattern)属于行为型模式,指对象有某个行为,但是在不同的场景中,该行为有不同的实...
      99+
      2023-06-20
    软考高级职称资格查询
    编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
    • 官方手机版

    • 微信公众号

    • 商务合作