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

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

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

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

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

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

目录

一、初识

二、案例

 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.mzph.cn/news/634159.shtml

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

相关文章

怎么做好小红书投放复盘,品牌运营总结

小红书作为主流的传播平台,聚焦了众多品牌的关注。但是对于小红书达人投放而言,是否完成投放就意味着任务结束了呢?其实并非如此,达人投放复盘也非常重要。今天我们就带大家了解一下怎么做好小红书投放复盘,品牌运营总结&#xf…

element中Table表格控件单选、多选功能进一步优化

目录 一、代码实现1、 父组件2、子组件&#xff08;弹框&#xff09; 二、效果图 一、代码实现 1、 父组件 <template><div><!-- 用户选择嵌套弹框 --><el-dialog :close-on-click-modal"false" :close-on-press-escape"false" tit…

零基础学Python(1)— 一文带你了解什么是Python(包括Python解释器安装步骤等)

前言&#xff1a;Hello大家好&#xff0c;我是小哥谈。从今天开始&#xff0c;我们就一起进入Python的世界&#xff01;&#x1f389;为了让大家能够牢固地掌握Python语言&#xff0c;本系列文章就循序渐进&#xff0c;从最基础的知识开始讲起&#xff0c;教大家如何去使用Pyth…

Spring-BeanPostProcessor PostConstruct init InitializingBean 执行顺序

执行顺序探究 新建一个对象用于测试 Component public class Student implements InitializingBean {private String name;private int age;public String getName() {return name;}public void setName(String name) {this.name name;}public int getAge() {return age;}pu…

代码之外:工程师的成长进阶秘籍

程序员只懂技术能行吗&#xff1f; 为什么说技术人员“说”和“写”总得擅长一个&#xff1f; 你以为的“关注结果”是真的结果吗&#xff1f; 从一线工程师跃升团队管理者一共分几步&#xff1f; 在不断变化的职场…

Linux粘滞位的理解,什么是粘滞位?

文章目录 前言如何理解&#xff1f;粘滞位的操作最后总结一下 前言 粘滞位&#xff08;Stickybit&#xff09;&#xff0c;或粘着位&#xff0c;是Unix文件系统权限的一个旗标。最常见的用法在目录上设置粘滞位&#xff0c;如此以来&#xff0c;只有目录内文件的所有者或者root…

zabbix其他配置

自动发现 zabbix server 主动的去发现所有的客户端&#xff0c;然后将客户端的信息登记在服务端上。 缺点是如果定义的网段中的主机数量多&#xff0c;zabbix server 登记耗时较久&#xff0c;且压力会较大。 systemctl disable --now firewalld setenforce 0 hostnamectl se…

还在手动复制文章吗?教你如何一键将文章从notion同步到WordPress

本文会给大家介绍如何在WordPress上安装一个插件&#xff0c;实现将notion上写的文章自动同步到WordPress上&#xff0c;从而提高写作效率&#xff0c;接下来请跟随我的脚步一起来操作吧&#xff01; 一、插件安装 在WordPress后台添加新插件页面中搜索“notion”&#xff0c;…

ip2domain - 批量查询ip对应域名、备案信息、百度权重

免责声明 由于传播、利用本文章所提供的信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;文章及作者不为此承担任何责任&#xff0c;一旦造成后果请自行承担&#xff01;如有侵权烦请告知&#xff0c;我们会立即删除并致歉。谢谢&#xf…

应该怎样保存用户密码

应该怎样保存用户密码&#xff1f; 首先&#xff0c;MD5 其实不是真正的加密算法。所谓加密算法&#xff0c;是可以使用密钥把明文加密为密文&#xff0c;随后还可以使用密钥解密出明文&#xff0c;是双向的。 使用 MD5 运算后得到的都是固定长度的摘要信息或指纹信息&#x…

C#:接口中如何将某个值类型的字段传null?

在实际对接第三方接口时&#xff0c;偶尔会有一些字段在某些情况下是不需要传值的。那如何处理呢&#xff1f; 有两种方法&#xff1a; 1、将值类型改为可空类型&#xff1b; 2、定义基类&#xff0c;基类包含所有必须要传的字段&#xff0c;子类则加入偶尔需要传的字段。 下…

java注释

注释 1、单行注释 2、多行注释 3、文档注释&#xff08;重点&#xff09; /*** author cx* version 1.0*/ public class Comment{//编写一个main方法public static void main(String[] args){System.out.println("hello,world~");} } 1. 在D盘找到javacode文件&a…

电梯节能落座-智慧停车场️,电梯不仅可载人也可以载汽车!

电梯不仅可载人也可以载汽车哦&#xff01; 在北京市丰台区&#xff0c;有这么一个智慧停车场&#x1f17f;️ &#xff0c;共298个停车位&#xff0c;全部智能一体化&#xff0c;简直是“豪华” “智能” 的象征。 523能源&#xff1a;小伍&#xff0c;你跑题了... 小伍&am…

【unity学习笔记】语音驱动blendershape

1.导入插件 https://assetstore.unity.com/packages/tools/animation/salsa-lipsync-suite-148442 1.选择小人&#xff0c;点击添加组件 分别加入组件&#xff1a; SALSA EmoteR Eyes Queue Processor&#xff08;必须加此脚本&#xff09;&#xff1a;控制前三个组件的脚本。…

Python GUI 新手入门教程:轻松构建图形用户界面

Python 凭借其简单性和多功能性&#xff0c;已经成为最流行的编程语言之一。被广泛应用于从 web 开发到数据科学的各个领域。 在本教程中&#xff0c;我们将探索用于创建图形用户界面&#xff08;GUIs&#xff09;的 Python 内置库&#xff1a; Tkinter&#xff1a;无论你是初…

golang 中使用 statik 将静态资源编译进二进制文件中

现在的很多程序都会提供一个 Dashboard 类似的页面用于查看程序状态并进行一些管理的功能&#xff0c;通常都不会很复杂&#xff0c;但是其中用到的图片和网页的一些静态资源&#xff0c;如果需要用户额外存放在一个目录&#xff0c;也不是很方便&#xff0c;如果能打包进程序发…

Byrdhouse AI实时语音翻译工具,可以在视频通话中翻译100多种语言

你是否曾经在跨国会议或与外国友人聊天时&#xff0c;因为语言不通而感到尴尬或困扰&#xff1f;是不是还在找可以实时翻译的软件或者APP&#xff1f;现在&#xff0c;有了这款语音翻译神器&#xff0c;一切都将变得简单&#xff01; 免费使用链接&#xff1a;https://byrdhous…

Text:焦点切换文字颜色随之改变

按Tab键切换2段文字的焦点&#xff0c;哪段文字的焦点为true&#xff0c;则字体颜色变为红色。 import QtQuickWindow {width: 640height: 480visible: truetitle: qsTr("2.2 属性")Rectangle {Text {id: thislabeltext: qsTr("hello world")font.pixelSiz…

龙芯3A5000上使用腾讯会议

原文链接&#xff1a;龙芯3A5000上使用腾讯会议 hello&#xff0c;大家好啊&#xff01;今天我要给大家介绍的是在龙芯3A5000处理器上安装使用腾讯会议的经验分享。随着远程工作和在线会议的普及&#xff0c;腾讯会议成为了许多人日常工作不可或缺的工具。而对于使用龙芯3A5000…

Open3D 点云转深度图像

目录 一、算法原理1、算法过程2、主要函数二、代码实现三、结果展示1、点云2、深度图像四、测试数据Open3D 点云转深度图像由CSDN点云侠原创。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。<