【设计模式】单例模式、“多例模式”的实现以及对单例的一些思考

文章目录

  • 1.概述
  • 2.单例模式实现代码
    • 2.1.饿汉式单例
    • 2.2.懒汉式单例
    • 2.3.双检锁单例
    • 2.4.静态内部类单例
    • 2.5.枚举单例
  • 3.对单例的一些思考
    • 3.1.是否需要严格的禁止单例被破坏?
    • 3.2.懒汉式真的比饿汉式更佳吗?
    • 3.3.单例存在的问题
  • 4.其他作用范围的单例模式
    • 4.1.线程内的单例
    • 4.2.进程间的单例
  • 5.“多例模式”
  • 6.总结

1.概述

单例模式是设计模式中最简单的一种,对于很多人来说,单例模式也是其接触的第一种设计模式,当然,我也不例外。这种设计模式在学习、面试、工作的过程中广泛传播,相信不少人在面试时遇到过这样的问题:“说说你最熟悉的集中设计模式”,第一个脱口而出的就是单例模式。

所谓的单例模式,就是在一定的作用范围内保证只有一个实例,这种模式,简单,但是想要使用好它,还需要学习一下它延伸出来的其他知识点,本篇博文就对单例模式做一下简单的整理,主要会包含以下几部分内容:

  • 单例模式的代码如何编写?
  • 是否需要严格的禁止单例被破坏?
  • 饿汉式和懒汉式应该如何选择?
  • 单例模式存在什么问题?
  • 线程内单例和进程间单例如何实现?
  • 什么叫做“多例模式”?

2.单例模式实现代码

单例模式的实现代码很多,下面会例举一些常见的方式。

Java中,单例模式的作用范围一般情况下指的是当前的Java进程,也就是进程内的对象保证唯一(当然还有线程内、进程之间的单例,下面会提到),所以我们需要保证实例只会被初始化一次,如何保证呢?

2.1.饿汉式单例

一个简单的做法,就是私有化构造方法,也就是不让外部的客户端对象来调用new方法,创建新的实例,而是在项目启动时,由单例类自行初始化,这就是饿汉式单例

/*** 饿汉式单例*/
public class HungrySingleton {private static final HungrySingleton HUNGRY_SINGLETON = new HungrySingleton();private HungrySingleton() {}public static HungrySingleton getInstance() {return HUNGRY_SINGLETON;}
}

2.2.懒汉式单例

如果不想再项目启动时初始化,而是在使用的时候再初始化对象,可以将对象的创建放到getInstance方法中,这种方式叫做懒汉式单例。这种方式在多线程的情况下会有线程安全问题,需要在创建对象时加锁。

/*** 懒汉式单例*/
public class LazySingleton {private static volatile LazySingleton lazySingleton;private LazySingleton() {}public static synchronized LazySingleton getInstance() {if (lazySingleton == null) {lazySingleton = new LazySingleton();}return lazySingleton;}
}

2.3.双检锁单例

在方法加的是类锁,一次只有一个线程可以获取到单例对象,为了提高获取对象的效率,取消在方法上的类锁,转而只给创建对象的那一行代码加锁。但是在并发的情况下,多个线程同时进入getInstance方法,都可以通过lazySingleton == null的判断,并在加锁那一行排队,每个线程都会创建一个新的对象,所以,我们需要在锁里面再判断一次对象是否创建。
这种在加锁的代码前后都进行一次相同判断的做法,我们叫做双重检查锁,简称:双检锁

/*** 双检锁单例*/
public class LazySingleton {private static volatile LazySingleton lazySingleton;private LazySingleton() {}public static LazySingleton getInstance() {if (lazySingleton == null) {synchronized (LazySingleton.class) {if (lazySingleton == null) {lazySingleton = new LazySingleton();}}}return lazySingleton;}
}

2.4.静态内部类单例

上面的懒汉式是为了保证懒加载的同时,又不能有线程安全问题,我们采用了加锁的方式,那么不加锁行不行呢?
当然是可以的,我们采用静态内部类在受访问时才会初始化的特性,来实现懒加载。

/*** 静态内部类单例*/
public class StaticClassSingleton {private StaticClassSingleton() {}public static StaticClassSingleton getInstance() {return InnerStaticClassSingleton.STATIC_CLASS_SINGLETON;}private static class InnerStaticClassSingleton {private static final StaticClassSingleton STATIC_CLASS_SINGLETON = new StaticClassSingleton();}
}

2.5.枚举单例

Java中的枚举是天然的单例模式,这种方式实现最简单,而且不会有线程安全问题,也能避免通过反射或者反序列化创建新的对象。
下面代码中的INSTANCE就是单例对象了。

/*** 枚举单例*/
public enum EnumSingleton {INSTANCE;
}

3.对单例的一些思考

3.1.是否需要严格的禁止单例被破坏?

在上面的代码例子中,我们采用的方式是私有化构造方法方法来避免外部对象new出新的单例对象,但这种方式并不能完全避免创建出新的对象。其实上面的枚举单例中已经提到了,可以通过反射、反序列化等方式,创建出新的对象。

在考虑如何避免通过反射、反序列化创建对象,可以先思考一下,有没有必要去避免?

还是回到最初那个点,我们用了这么多方式来实现单例模式,最终的目的就是为了在进程内的作用范围内只有一个实例。而在代码的开发过程中,我们做好规范和约束,以人为的方式来控制对象的创建数量,哪怕没有私有化构造方法方法也能保证单例。而我们私有化构造方法,更多的是给开发者做出一个提示,这个类是个单例类,并且防止一定的误操作。
可以想象一下,我们要完全将各类创建和初始化的逻辑都“封闭”掉,代码会臃肿到什么地步,而这部分“封闭”的逻辑在业务开发中我们完全使用不到,所以更建议以一种“约定由于配置”的方式来处理单例的创建问题。

综上,做到私有化构造方法这一步就够了,不需要过度开发。

3.2.懒汉式真的比饿汉式更佳吗?

懒汉式主要是为了做懒加载,当单例对象没有使用的时候就不创建和初始化,特别是初始化是需要加载的资源比较多、比较耗时的时候,用懒加载可以加快项目启动的速度,同时又能减少系统的资源浪费。
但是从另一个角度讲,如果不是在启动的时候初始化,那就是在客户端调用的时候初始化,想象一下一个高并发的互联网项目,如果在客户端调用的时候再做耗时的初始化动作,就可能造成接口的请求时间过长,接口超时等,会影响一批用户的使用体验。
另外,如果初始化的过程中存在一些异常情况,我们应该让问题在项目启动时就暴露出来,及时修复,而不是在用户使用的时候才暴露问题。

综上,在业务开发中或许饿汉式单例是更好的选择。

3.3.单例存在的问题

单例模式编写和使用都很简单,但是它也存在一些问题,例如:

  • 面向对象支持不好:单例模式在作用范围内只有一个实例,那就无法通过创建更多的实例来使用面向对象的抽象、继承、多态等特性,更像是一种面向过程的写法。
  • 违反开闭原则:无法拓展,每一次迭代都需要修改原有的代码。
  • 违反单一职责:单例模式既创建对象、又管理对象,职责模糊,可能会导致代码变得复杂。

综上,单例模式存在一定的问题,如果存在拓展的需求就尽可能的避免使用单例模式。

但如果在不需要大量的拓展,又没有业务间的复杂依赖关系,使用单例模式就比较简洁方便也不失为一种选择,例如各种无状态的工具类。

4.其他作用范围的单例模式

4.1.线程内的单例

即单例对象在线程内时唯一的,线程之间不是唯一的,我们开发中有一种很常见的情况:线程局部变量,一般是通过ThreadLocal来做的。
实现原理也比较简单,其实就是使用一个全局的Map来保存对象,以线程对象threadkey,以需要保存的单例对象为value,这样就保证了一个线程只对应一个对象。
在业务流程中的用户登录信息,往往就是保存在ThreadLocal中的,另外PageHelper这个著名的工具类也是通过ThreadLocal来实现的。


如果想了解ThreadLocal的使用方式,可以参考我的另一篇博客《【并发编程】(九)线程安全的代码及ThreadLocal的使用》
如果想了解它详细的实现原理,可以参考《【并发编程】(十)线程局部变量——ThreadLocal原理详解》

4.2.进程间的单例

进程间的单例,更常用的一种说法分布式环境中的单例,这类需求我们使用的也比较多,其实现原理也比较简单,就是将单例对象通过序列化的方式存储在一个多个服务都会共同访问的存储区域中,例如一个共享的文件中、一些分布式的中间件中,例如rediszk等等,而最常见的当然就是分布式锁
我们只需要为单例对象创建出一个唯一标识,在每个服务中判断唯一标识是否存在即可。

5.“多例模式”

多例模式是单例模式中的一种特例,即可以在一定数量范围内创建类的多个实例,还有一层理解就是不同类型的对象可以创建多个,想通类型的对象只能创建一个,后者的概念使用的更多。

以日志打印为例,我们引入Slf4J后通过下面的方式获得一个日志对象:

private Logger logger = LoggerFactory.getLogger(xxx.class);

这里获取的logger如果后面的class对象相同,获取的就是同一个对象,这种方式更像是工厂模式,在代码中看到的也是工厂模式,如下图:
我们进入这个工厂模式的方法后,可以看到下面的代码:
在这里插入图片描述
这里就非常明显了,这就是一种单例模式的创建方式,通过一个Map将单例对象管理起来,如果Map中有就直接返回,如果没有就创建一个并放入到Map中,这里的对象都是logger对象,只是使用日志的类不一样,这就是多例模式的一种体现。

另外,在Spring中如果配置的bean是单例的,其创建方式也与这种方式类似。

6.总结

本来主要讲述了以下几个点:

  • 单例模式的编写方式
    饿汉式、懒汉式、静态内部类、枚举
  • 是否需要严格的禁止单例被破坏:
    没有必要写的太严格,可以通过规范的方式来约束
  • 饿汉式和懒汉式应该如何选择:
    让耗时操作提前初始化,让问题提早暴露,及时修改,而不是让用户去发现
  • 单例模式存在什么问题
    没有面向对象,拓展性差
  • 线程内单例和进程间单例如何实现
    线程内单例通过线程局部变量来实现,进程间的单例通过共享的存储区域来实现
  • 什么叫做“多例模式”
    在一定数量范围内可以创建多个,或者不同的类可以有多个、相同的类只能有一个

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

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

相关文章

关闭VS Code中的鼠标悬停时的提示框(MDN Reference)

在使用VS Code编辑器写html文件时,鼠标悬停在写的某些内容时会弹出一个提示框,如下图: 这个提示是比较烦人的,接下来分享关闭它的教程: 这里是以Win10版的Visual Studio Code为例 1.打开VS Code 的设置界面 2.在扩展…

《开箱元宇宙》:《福布斯》如何通过 Web3 改进讲故事的方式

你们是否想知道 The Sandbox 如何融入世界上最具标志性的品牌和名人的战略?在本期《开箱元宇宙》系列中,我们与《福布斯》一起探讨了他们为何决定在 The Sandbox 中尝试 Web3,以及他们如何改变讲故事的方式,以便在一次体验中吸引超…

字节跳动2023测试开发岗 3+1 面经+经验分享(收到offer,入职月薪27K)

现在,招聘黄金时间已经来临,在网上看了很多大佬的面经,也加了很多交流群,受到了很多朋友的提点,今天终于轮到我来分享面经啦,之前面试了几家公司,最后在十月初拿到了字节跳动测试岗的 offer&…

口袋参谋:如何写出高权重标题?用对这招很重要!

​如何写出高权重标题?这是99.99%的卖家都存在的疑虑! 以前写高权重标题,很多卖家往往会复制同行竞品爆款标题到淘宝首页搜索框,然后在全标题后面加上几个字母,就可以拆分爆款标题。 这个问题我之前也说过&#xff0…

【GO入门】环境配置及Vscode配置

1 GO环境配置 欢迎来到Go的世界,让我们开始探索吧! Go是一种新的语言,一种并发的、带垃圾回收的、快速编译的语言。它具有以下特点: 它可以在一台计算机上用几秒钟的时间编译一个大型的Go程序。Go为软件构造提供了一种模型&…

【C++】:string用法详解

朋友们、伙计们,我们又见面了,本期来给大家解读一下有关Linux的基础知识点,如果看完之后对你有一定的启发,那么请留下你的三连,祝大家心想事成! C 语 言 专 栏:C语言:从入门到精通 数…

使用PyTorch解决多分类问题:构建、训练和评估深度学习模型

💗💗💗欢迎来到我的博客,你将找到有关如何使用技术解决问题的文章,也会找到某个技术的学习路线。无论你是何种职业,我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章,也欢…

Vue配置全局变量config.js

Vue配置全局变量config.js 若config.js在public目录下 在index.html中引入 这样配置是为了防止路由前缀&#xff0c;如果直接“/config.js”&#xff0c;若路由没有前缀还好&#xff0c;要是有就需要配置为“<% BASE_URL %>config.js”

Xilinx IP 10G Ethernet PCS/PMA IP Core

Vivado 10G Ethernet PCS/PMA介绍 1介绍 完整的10G以太网接口如下图,分为10G PHY和10G MAC两部分。 这篇文章重点讲 10G Ethernet PCS/PMA。 2 IP的基本介绍 10G以太网物理编码子层/物理介质连接(PCS/PMA)核心在Xilinx 10G以太网介质访问控制器(MAC)核心和具有10Gb/s…

Linux网络编程系列之UDP组播

Linux网络编程系列 &#xff08;够吃&#xff0c;管饱&#xff09; 1、Linux网络编程系列之网络编程基础 2、Linux网络编程系列之TCP协议编程 3、Linux网络编程系列之UDP协议编程 4、Linux网络编程系列之UDP广播 5、Linux网络编程系列之UDP组播 6、Linux网络编程系列之服务器编…

尚硅谷Flink(一)

目录 ☄️前置工作 fenfa脚本 &#x1f30b;概述 ☄️Flink是什么 ☄️特点&#xff08;多nb&#xff09; ☄️应用场景&#xff08;不用看&#xff09; ☄️分层API &#x1f30b;配环境 ☄️wordcount ☄️WcDemoUnboundStreaming &#x1f30b;集群部署 ☄️集…

xml的语法

<!-- 1、每一个xml,有且只有一个根标签&#xff0c;所有xml标签必须写在根标签中 2、标签必须要有合闭 3、xml格式是否正确&#xff0c;可以通过浏览器打开xml。来校验xml格式是否正确 4、xml是区别大小写的 5、xml书写标签名时&#xff0c;不要出现空格等特…

微信小程序的框架

目录 一、视图层 1. WXML 数据绑定 列表渲染 条件渲染 模板 2. WXSS 尺寸单位 样式导入 内联样式 选择器 3. WXS事件 二、逻辑层 1. 页面生命周期 2.跳转 1. 一级跳一级 2. 一级跳二级 3. 二级跳二级 4. 二级跳一级 总结 带给我们的收获 一、视图层 1. …

idea 启动项目报错 Command line is too long

1.idea 启动报错 Command line is too long&#xff0c;启动报错信息&#xff1a;Error running ‘Application‘: Command line is too long. 2.如何解决&#xff1f; 1&#xff09;idea打开一个项目。 2.打开项目下的*.idea* 文件夹下的 workspace.xml 文件。 3.在<co…

springboot 导出word模板

一、安装依赖 <dependency><groupId>com.deepoove</groupId><artifactId>poi-tl</artifactId><version>1.12.1</version></dependency>二、定义工具类 package com.example.springbootmp.utils;import com.deepoove.poi.XWP…

5.Python-使用XMLHttpRequest对象来发送Ajax请求

题记 使用XMLHttpRequest对象来发送Ajax请求&#xff0c;以下是一个简单的实例和操作过程。 安装flask模块 pip install flask 安装mysql.connector模块 pip install mysql-connector-python 编写app.py文件 app.py文件如下&#xff1a; from flask import Flask, reque…

好用的跨平台同步笔记工具,手机和电脑可同步的笔记工具

在这个快节奏的工作环境中&#xff0c;每个人都在寻找一种方便又高效的方式来记录工作笔记。记录工作笔记可以帮助大家统计工作进展&#xff0c;了解工作进程&#xff0c;而如果工作中常在一个地方办公&#xff0c;直接选择电脑或者手机中笔记工具来记录即可&#xff0c;但是对…

[部署网站]01安装宝塔面板搭建WordPress

宝塔面板安装WordPress&#xff08;超详细&#xff09;_Wordpress主题网 参考教程 宝塔面板 - 简单好用的Linux/Windows服务器运维管理面板 官网 1.首先你需要一个服务器或者主机 &#xff08;Windows系统或者Linux系统都可以&#xff09; 推荐Linux系统更稳定&#xff0c;…

Axure RP医疗在线挂号问诊原型图医院APP原形模板

医疗在线挂号问诊Axure RP原型图医院APP原形模板&#xff0c;是一款原创的医疗类APP&#xff0c;设计尺寸采用iPhone13&#xff08;375*812px&#xff09;&#xff0c;原型图上加入了仿真手机壳&#xff0c;使得预览效果更加逼真。 本套原型图主要功能有医疗常识科普、医院挂号…

海思平台SS528V100编译 linux kernel tun.c eth_get_headlen 编译 出错的问题

osdrv目录下 make kernel 会去opensource目录下解压linux内核压缩包 同时打上很多补丁 如上图 查看Makefile 如下 有打补丁的命令 然后 然后我们的应用程序用到一个特性 需要打开tun/tab这两个属性 打开之后编译内核出错 查到最后发现 没打补丁之前的文件 没问题 …