探索设计模式的魅力:“感受单例模式的力量与神秘” - 掌握编程的王牌技巧

news2024/9/29 15:33:49

     在软件开发的赛场上,单例模式以其独特的魅力长期占据着重要的地位。作为设计模式中的一员,它在整个软件工程的棋盘上扮演着关键性角色。本文将带你深入探索单例模式的神秘面纱,从历史渊源到现代应用,从基础实现到高级技巧,经过戏剧性的转折和层层推进,我们将一步步揭开这一模式背后的秘密。

     首先,文章将串起时间的线索,带你重回单例模式的起源,理解它在软件工程历史中的地位。经过时间的流逝,单例模式不仅保持了其原有的魅力,而且随着新的编程语言和技术的发展,还展示出了新的活力和应用场景。

     接着,细致解码单例模式的基础结构,为你展现其底层的实现方式。通过对不同实现策略的探讨,我们会评估各自的优势和潜在风险,帮助你作出明智的设计选择。无论是饿汉式的立即加载,还是懒汉式的延迟兴建,或是静态内部类和枚举方式的现代演化,本文将在详尽的案例分析中,为你搭建一座走向单例模式深入理解的桥梁。

     文中还将精心打造实战场景,凸显单例模式在不同应用中的策略运用。从多线程的并发控制到资源访问的优化,从全局状态管理到服务类的持久化,单例模式如何巧妙地解决现实问题——这些都将在本文中一一揭晓。

     在“单挑编程世界”的冒险中,你将体验到单例模式的实践力量。面对代码的战场,这篇文章将成为你的秘密武器,助你披荆斩棘,挖掘单例模式背后藏匿的珍宝。无论你是设计模式的新手,还是寻求更高级理解的资深开发者,本文都将与你并肩作战,一起探索编程世界的深奥与精彩。

目录

一、初识

二、案例

 2.1 分析与实现-懒汉式

 2.2 分析与实现-静态内部类

 2.3 分析与实现-饿汉模式

 2.4 分析与实现-双重检查锁

 2.5 分析与实现-枚举

三、讲解

 3.1 功能

 3.2 范围

 3.3 调用顺序示意图

 3.4 优缺点

四、练习


一、初识

     单例模式的历史渊源可以追溯到上世纪90年代,当时设计模式的概念由Erich Gamma、Richard Helm、Ralph Johnson和John Vlissides等人在《设计模式:可复用面向对象软件的基础》一书中首次提出。

     最初的设计模式书籍中并没有明确提到单例模式,但许多后来的书籍和文章将其作为设计模式的一部分进行讨论和描述。单例模式的概念是为了解决在软件开发中需要创建一个全局唯一对象的问题。

     随着编程语言和技术的发展,单例模式逐渐成为软件工程领域的重要模式之一,并广泛应用于各种类型的应用程序和系统中。它在许多领域具有实际应用,包括但不限于以下几个方面:

     1. 资源管理单例模式常用于管理共享资源,例如数据库连接池、线程池、日志管理器等。通过使用单例模式,可以确保在整个应用程序中只有一个实例存在,从而有效地管理资源和提高性能。

     2. 配置管理单例模式在配置管理中也扮演着重要角色。例如,一个应用程序可能需要读取配置文件中的信息,并在整个应用程序中共享这些配置信息。通过使用单例模式,可以确保在整个应用程序生命周期中只有一个配置对象存在。

     3. 全局状态管理在某些情况下,需要在应用程序中维持一些全局状态数据,例如游戏中的玩家信息、应用程序的运行时状态等。通过使用单例模式,可以方便地访问和更新这些全局状态数据,并确保数据的一致性和唯一性。

     4. 日志记录单例模式常用于日志记录器的实现。通过使用单例模式,可以确保在整个应用程序中只有一个日志记录器实例,从而统一管理和记录日志信息。

     5. 缓存管理在需要缓存数据的场景中,单例模式可以用来实现缓存管理器。通过单例模式,可以确保整个应用程序中只有一个缓存管理器实例,提高缓存的效果和一致性。

     6. 节约内存和提高性能由于单例模式只创建一个实例,可以减少内存占用,并且在运行时能够提供更高的性能。这在需要频繁创建和销毁对象的场景中尤为重要,如线程池、网络连接等。

     7. 限制实例化和控制访问单例模式可以限制一个类只能创建一个实例,并提供一个统一的访问点。这可以防止其他对象直接实例化该类,从而增加了对实例化过程的控制和安全性。

     8. 模块化和解耦合单例模式可以帮助实现模块化和解耦合,在应用程序中以模块的形式组织和管理代码。单例模式将整个模块封装在一个类中,对外暴露统一的接口,使得模块之间的交互更清晰和简化。

        

二、案例

     场景:银行推出一个新的产品(就叫新产品了),通过获取个人缴纳部分的公积金额度乘以某倍数得到的总金额来作为准入的其中一个规则和计算授信额度的标准。

        ...

     最终,就是调某个接口(叫外部产品了)来获取客户的公积金信息。外部产品SDK说明第一步是实例一个Client,由Client负责组装数据、加密(解密)等然后发post请求来获取公积金信息。        

        

 2.1 分析与实现-懒汉式

     考虑到是尝试新产品和部分地区尝试上线、硬件软件、服务器性能情况等综合因素考虑选择使用懒汉式来实例化产品Client。

     实现示例代码:

public class LazySingleton {
    private static SdkClient sdkClient;

    private LazySingleton() {
        // 私有构造函数
    }
    public static synchronized SdkClient getInstance() {
        if (sdkClient == null) {
            sdkClient = new SdkClient();
        }
        return sdkClient;
    }
}

     在懒汉模式中,构造函数是私有的,确保外部无法直接实例化该类。getInstance() 方法返回单例对象的实例,并在首次调用时创建对象。 

     优点:

  • 延迟加载,最大限度地节省资源和提高性能。
  • 只有在需要时才会创建实例,适用于大型对象或高开销资源的情况。
  • 线程安全:通过同步处理来解决多线程环境下的竞态条件。

     缺点:

  • 需要考虑线程安全问题,可能需要使用同步机制,如使用synchronized关键字或通过双重检查锁定等。
  • 同步机制会增加额外的开销,可能会影响性能。
  • 可能存在序列化和反射攻击的漏洞,需要做相应的处理来防止。

         

 2.2 分析与实现-静态内部类

     静态内部类方式利用Java的类加载机制来实现懒汉式的延迟加载。

     实现示例代码:

public class StaticInnerSingleton {
    private StaticInnerSingleton() {
        // 私有构造函数
    }

    private static class SingletonHolder {
        private static final SdkClient instance = new SdkClient();
    }

    public static SdkClient getInstance() {
        return SingletonHolder.instance;
    }
}

     在这个示例中,私有的构造函数确保了外部无法直接实例化该类的对象。通过静态内部类SingletonHolder来持有单例对象,并且该类的实例化操作是在静态初始化阶段(即类加载时)完成的,保证了线程安全性。

     优点:

  1. 懒加载:单例对象的实例化在调用getInstance()方法时才会执行,实现了延迟加载的效果。

  2. 线程安全:由于静态内部类的特性,当多个线程同时访问getInstance()方法时,静态内部类的实例化操作只会执行一次,保证了线程安全性。

  3. 简洁性:相对于双重检查锁等方式,使用静态内部类实现单例模式的代码更加简洁明了。

     缺点:

  1. 需要了解并理解静态内部类的工作原理:虽然使用静态内部类实现单例模式代码相对简洁,但是理解这种实现方式的工作原理可能需要一定的知识和经验。对于不熟悉静态内部类的开发者来说,可能需要花费一些时间来理解其背后的概念和实现细节。

  2. 不支持传递参数的实例化:静态内部类实现的单例模式无法直接传递参数给单例对象的构造函数,因为静态内部类的实例化是在类加载时完成的。如果需要传递参数的实例化,可能需要使用其他方式实现。

  3. 需要额外的类:使用静态内部类实现单例模式需要额外定义一个静态内部类来持有单例对象,这增加了代码中的类数量。虽然这对于代码的组织和结构有一定的好处,但对于一些简单的应用场景来说可能显得有些冗余。

  4. 序列化和反序列化的处理:如果单例对象需要支持序列化和反序列化操作,需要额外处理,否则在反序列化时会得到不同的实例。可以通过实现readResolve()方法来解决这个问题。

        

 2.3 分析与实现-饿汉模式

     因为支行的各种软件产品的建设都要报备到省总部,由于历史某种原因,以前报软件产品比较少。所以好多产品都集成一个应用上,一个应用几乎每有新功能需要上线(这种频繁上线方式肯定是不建议的,但有时候又是非常必须的),导致某天第一个客户经理办业务时失败,查原因是实例化和发post请求超时了(内网向外网发请求,网络上做了好多层的代理转发)。小白就想着,那就修改一下外部产品Client的实例化方式吧。

     痛点:第一个应用加载太慢导致业务办理失败。

     实现示例代码:

public class EagerSingleton {
    private static final SdkClient sdkClient = new SdkClient();
    
    private EagerSingleton() {
        // 私有构造函数
    }
    public static SdkClient getInstance() {
        return sdkClient;
    }
}

     在饿汉模式中,单例实例在类加载时就被创建,并且是静态的、final的常量。getInstance() 方法直接返回该实例,而无需进行额外的创建过程。

     由于在类加载时就创建了单例实例,因此在多线程环境中是线程安全的,不需要额外处理同步问题。

     优点:

  • 线程安全:由于在类加载时就创建实例,因此不会存在多线程并发访问创建实例的问题,无需考虑同步和线程安全。
  • 简单明了:代码相对较为简单,不需要额外的同步处理,逻辑清晰。

     缺点:

  • 资源浪费:在某些情况下,如果单例对象的创建和初始化比较耗时,而且在程序的整个生命周期中可能并不会被立即使用,就会造成资源的浪费。
  • 引入不必要的依赖:饿汉模式可能会在应用启动时加载大量实例,增加了启动时间,还可能引入不必要的依赖关系。

        

 2.4 分析与实现-双重检查锁

     有段时间发现服务器出现频繁的GC,排查发现大量的实例使用比较少但是又不能没有,其中就有外部产品Client对象。最近服务器和网络都做了调优,再加上产品已大量推广使用,在某一些高峰期可会存在大量并发的情况等等种种原因综合考虑。小白就想着,那就修改一下外部产品Client的实例化方式吧。用双重检查锁来实现实例化外部产品Client对象。

     实现示例代码:

public class DoubleCheckedSingleton {
    private volatile static SdkClient sdkClient;

    private DoubleCheckedSingleton() {
        // 私有构造函数
    }

    public static SdkClient getInstance() {
        if (sdkClient == null) {
            synchronized (SdkClient.class) {
                if (sdkClient == null) {
                    sdkClient = new SdkClient();
                }
            }
        }
        return sdkClient;
    }
}

     在双重检查锁模式中,通过在getInstance()方法中使用双重检查来确保只有在实例为null时才创建新的实例。

  • 第一次检查:检查实例是否已经创建,如果没有创建过,则进入同步块。
  • 第二次检查:在同步块中再次检查实例是否已经创建,这是为了避免在多个线程同时通过第一次检查的情况下,都进入同步块创建实例,从而造成多次实例化的问题。

     为了确保线程之间的可见性,需要使用volatile关键字修饰instance变量。volatile关键字的作用是禁止指令重排,保证每个线程都能正确看到instance变量的最新值,避免在多线程环境下出现问题。

     优点:

  • 延迟加载:双重检查锁可以在需要时才创建实例,避免了一开始就创建实例造成的资源浪费。
  • 提高性能:只有在第一次获取实例时才需要同步,后续获取实例时无需进行同步,提高了性能。

     缺点:

  • 使用双重检查锁实现单例模式仍然需要谨慎对待,确保代码正确性,代码的复杂性增加了维护和理解的难度,可能会导致潜在的bug。

        

 2.5 分析与实现-枚举

     实现示例代码:

public enum EnumSingleton {
    INSTANCE;

    private int value;

    EnumSingleton() {
        // 构造函数可以添加初始化逻辑
        value = 10;
    }

    public int getValue() {
        return value;
    }
}

     在这个示例中,EnumSingleton是一个枚举类,其中INSTANCE是唯一的枚举常量,代表了单例对象。在构造函数中,我们可以添加额外的初始化逻辑。在这里,我们将value初始化为10。

     假设在其他地方调用时,可以通过EnumSingleton.INSTANCE来获取初始对象,并调用其方法:

int value = EnumSingleton.INSTANCE.getValue();
System.out.println(value);  // 输出:10

     通过枚举类的常量INSTANCE获取到初始对象,然后可以调用其方法和访问其成员变量。 

     优点:

  • 线程安全:枚举实现的单例模式在创建实例时是线程安全的。枚举常量的实例化在类加载时完成,保证了全局只有一个实例,并且在多线程环境下也是安全的。
  • 防止反序列化创建新对象:枚举类默认实现了Serializable接口,并且枚举对象的反序列化操作不会创建新的对象。这使得枚举单例在涉及到序列化和反序列化的场景中更加安全。
  • 简单明了:使用枚举实现单例模式代码简洁明了,不需要编写复杂的线程安全逻辑或者使用双重检查锁等方式,只需声明一个枚举常量即可。

     缺点:

  • 不能延迟加载:枚举实现的单例模式在类加载时就完成了实例化,因此无法实现延迟加载的需求。如果应用在初始化时对资源消耗较大,无法延迟加载可能会影响应用的性能。
  • 有限的扩展性:枚举常量在枚举类中是固定的,无法在运行时动态地添加额外的枚举常量。这意味着枚举单例模式的扩展性相对受限,无法通过添加更多的实例来应对不同的需求。

        

三、讲解

     单例模式的本质:控制实例数目。

 3.1 功能

     单例模式是用来保证这个类在运行期间只会被创建一个类实例,另外,单例模式还提供了一个全局唯一访间这个类实例的访问点,就是getlnstance 方法。不管采用懒汉式还是饿汉式的实现方式,这 个全局访问点是一样的。

    对于单例模式而言,不管采用何种实现方式,它都是只关心类实例的创建问题,并不关心具体的业务功能。

        

 3.2 范围

     目前Java 里面实现的单例是一个虚拟机的范围。因为装载类的功能是虚拟机的,所以 一个虚拟机在通过自己的ClassLoader 装载饿汉式实现单例

类的时候就会创建一个类的实例。

     这就意味着如果一个虚拟机里面有很多个ClassLoader,而且这些ClassLoader 都装载某个类的话,就算这个类是单例,它也会产生很多个实例。当然,如果一个机器上有多个虚拟机的话,那么每个虚拟机里面都应该至少有一个这个类的实例,也就是说整个机器上就有很多个实例,更不会是单例了。

        

 3.3 调用顺序示意图

28ed1e5193c9427b921aa89335f9bfcf.png

         

 3.4 优缺点

     1. 时间与空间

  • 比较懒汉式和饿汉式:懒汉式是典型的时间换空间,也就是每次获取实例都会进行判断,看是否需要创建实例,浪费判断的时间。当然,如果一直没有人使用的话,那就不会创建实例,则节约内存空间。
  • 饿汉式是典型的空间换时间,当类装载的时候就会创建类实例,不管你用不用,先创建出来,然后每次调用的时候,就不需要再判断 了,节省了运行时间。

     2. 线程安全

  • 从线程安全性上讲,不加同步的懒汉式是线程不安全的。
  • 饿汉式是线程安全的,因为虚拟机保证只会装载一次,在装载类的时候是不会 发生并发的。
  • 如何实现懒汉式的线程安全,懒汉式也是可以实现线程安全的,只要在getInstance()方法上加上synchronized 即可。加上synchronized会降低整个访问的速度,而且每次都要判断。
  • 双重检查加锁,可以使用“双重检查加锁” 的方式来实现,就可以既实现线程安全,又能够使性能不受到很大的影响:并不是每次进入getlnstance 方法都需要同步,而是先不同步,进入方法过后,先检查实例是否存在,如果不存在才进入下面的同步块这是第一重检查。进 入同步块过后,再次检查实例是否存在,如果不存在,就在同步的情况下创建一个实例,这是第二重检查。这样一来,就只需要同步一次了,从而减少了多次在同步情况下进行判断所浪费的时间。双重检查加锁机制的实现会使用一个关键字volatile,它的意思是:被volatile 修饰的变量的值,将不会被本地线程缓存,所有对该变量的读写都是直接操作共享内存,从而确保多个线程能正确的处理该变量。

        

四、练习

     请编写 Triple类,实现最多只能生成3个 Triple类的实例,实例编号分别为 0 , 1 , 2且可以通过 getInstance(int id)来获取该编号对应的实例。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1396088.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数据结构实验6:图的应用

目录 一、实验目的 1. 邻接矩阵 2. 邻接矩阵表示图的结构定义 3. 图的初始化 4. 边的添加 5. 边的删除 6. Dijkstra算法 三、实验内容 实验内容 代码 截图 分析 一、实验目的 1.掌握图的邻接矩阵的存储定义; 2.掌握图的最短路径…

Idea 开发环境不断切换git代码分支导致冲掉别人代码

问题分析 使用git reflog查看执行命令,以下是发生事故的切换和提交动作 46f72622e1 HEAD{41}: commit: feat: 【Sales - 6.3】小程序端不登录也可以录入客户线索 c5e7d9f6e1 HEAD{42}: fetch origin feature/20240102_Sales6.3_xingang:feature/20240102_Sales6.3…

基于 IDEA 进行 Maven 依赖管理

一、依赖管理概念 Maven 依赖管理是 Maven 软件中最重要的功能之一。Maven 的依赖管理能够帮助开发人员自动解决软件包依赖问题,使得开发人员能够轻松地将其他开发人员开发的模块或第三方框架集成到自己的应用程序或模块中,避免出现版本冲突和依赖缺失等…

旧衣物回收小程序搭建,旧衣回收利润有多高?

在当下人们的生活水平日益提高,淘汰的旧衣物逐渐增加。面对这些旧衣物,很多人舍不得丢弃,既浪费又污染环境,只能常年堆积在家里。为了解决这一问题,旧衣回收成为了大众的选择。大家能够将淘汰的衣物进行回收&#xff0…

Java-NIO篇章(4)——Selector选择器详解

Selector介绍 选择器(Selector)是什么呢?选择器和通道的关系又是什么?这里详细说明,假设不用选择器,那么一个客户端请求数据传输那就需要建立一个连接,为了避免线程阻塞,那么每个客…

微分方程(1)微分方程的历史

总之就是学习吧~ Introduction Without knowing something about differential equations and methods of solving them, it is difficult to appreciate the history of this important branch of mathematics. Further, the development of differential equations is inti…

如何用GOWIN创建FPGA工程

高云FPGA如何创建工程 第一步:安装Gowin软件,这个在高云的官网是可以下载的 第二步:点开这个软件,点击“New Project…” 点击“Next” 选择自己对应的器件型号: 工程创建成功,如下图: 最后将…

给程序加个进度条吧!1行Python代码,快速搞定~

你在写代码的过程中,有没有遇到过以下问题? 已经写好的程序,想看看程序执行的进度? 在写代码批量处理文件的时候,如何显示现在处理到第几个文件了? 👆如上图所示的进度条是一个最好的解决方法…

如何解决分支机构无法连入总部采购管理系统的难题

案例背景: 某企业业务规模不断壮大,内部采购流程越发复杂,供应商资质情况各异难以管理,为提高内部采购效率和采购品质,优化供应链管理,确保采购环节公正透明可溯,该企业集中化部署了采购管理系…

高级RAG(八): 自动合并检索(Auto-merging Retrieval)

自动合并检索(Auto-merging Retrieval)是LlamaIndex的另外一种高级RAG技术,它有点类似与我们之间介绍的从小到大的检索,不过自动合并检索要比“从小到大的检索”稍微复杂一些,它首先将文档按一定的层次结构进行切割,然后在检索的时…

北斗卫星:助力社区矫正人员追踪与管理的科技突破

北斗卫星:助力社区矫正人员追踪与管理的科技突破 社区矫正人员是一个重要的社会群体,他们的安全和管理对于社会的和谐稳定至关重要。随着技术的飞跃发展,北斗卫星系统作为我国自主研发的卫星导航系统,正逐渐在社区矫正工作中发挥…

UE5 播放rtsp监控视频

1. 插件下载 https://github.com/inveta/InVideo https://github.com/inveta/InVideo/releases https://download.csdn.net/download/qq_17523181/88760489?spm1001.2014.3001.5501 插件目前支持5.1 / 5.0 2. 建立C UE5项目 重要:此插件支持C项目,不然不…

Windows Server搭建DNS服务器

服务器规划 DNS服务器IP为:192.168.2.82 需要自定义域名解析 192.168.2.82 dns.zhangsan.com 192.168.2.181 www.zhangsan.com 192.168.2.182 test1.zhangsan.com 192.168.2.183 test2.zhangsan.com 0. 服务器系统信息 1. 服务器初始化 1.1 查询IP 在命令行窗口中…

【React基础】– JSX语法

文章目录 认识JSX为什么React选择了JSXJSX的使用 React事件绑定this的绑定问题事件参数传递 React条件渲染React列表渲染列表中的key JSX的本质createElement源码Babel官网查看直接编写jsx代码 虚拟DOM的创建过程jsx – 虚拟DOM – 真实DOM声明式编程 阶段案例练习 认识JSX ◼ …

【重学C语言】四、常量、变量和基本数据类型

【重学C语言】四、常量、变量和基本数据类型 1. 常量和变量1.1 常量1.1.1 直接常量1.1.1.1 整数常量1.1.1.2 浮点常量1.1.1.3 字符常量1.1.1.4字符串常量 1.1.2符号常量 1.2 标识符1.3 变量1.3.1 变量的定义1.3.2 左值(Lvalues)和右值(Rvalue…

a-range-picker获取开始和结束时间

效果图 代码块 <a-range-picker change"onChangeRangeDate" format"YYYY-MM-DD"/>onChangeRangeDate(value,dateString){this.startDatedateString[0]this.endDatedateString[1]}

抓包工具Fiddler的下载、安装、配置、基本使用

文章目录 前言一、Fiddler下载二、Fiddler安装三、打开Fiddler四、Fiddler显示菜单栏及菜单栏功能基本介绍五、Fiddler的作用六、Fiddler的工作原理七、Fiddler基本功能使用1.导包2.规则设置Rules&#xff1a; 八、Fiddler设置抓取HTTPS的包九、抓取APP包必须设置十、Fiddler过…

谷达冠楠科技:抖音开网店新手小白可以卖的产品

随着互联网的发展&#xff0c;越来越多的人选择在网上开设自己的店铺。而抖音作为目前最火的短视频平台&#xff0c;也提供了开店的功能。那么&#xff0c;对于新手小白来说&#xff0c;抖音开网店可以卖哪些产品呢? 我们可以考虑的是服装类商品。抖音上有很多时尚博主&#x…

极致画质与流畅播放的完美结合,只在ProVideoPlayer for Mac!

ProVideoPlayer for Mac 是一款功能强大的专业级视频播放软件&#xff0c;旨在提供出色的用户体验和无与伦比的功能。以下是它的一些主要功能介绍&#xff1a; 多格式兼容&#xff1a;ProVideoPlayer for Mac 支持广泛的视频格式&#xff0c;包括常见的MP4、AVI、MOV&#xff0…

鸿蒙开发(五)鸿蒙UI开发概览

从用户角度来讲&#xff0c;一个软件拥有好看的UI&#xff0c;那是锦上添花的事情。再精确的算法&#xff0c;再厉害的策略&#xff0c;最终都得通过UI展现给用户并且跟用户交互。那么&#xff0c;本篇一起学习下鸿蒙开发UI基础知识&#xff0c;认识下各种基本控件以及使用方式…