当前位置: 首页 > news >正文

设计模式-单例模式实战

目录

  • 一、引言
  • 二、适用场景
  • 三、代码实战
    • 饿汉式单例模式
    • 懒汉式单例模式
    • 双重检查锁定单例模式
    • 静态内部类单例模式
  • 四、实际应用举例
      • Runtime解析
  • 五、结论

一、引言

    单例模式是一种创建型设计模式,用于确保一个类只有一个实例,且提供全局访问点以访问该实例。以下是单例模式的关键特点:

  • 私有构造函数:单例类通常会将其构造函数声明为私有,以防止外部直接创建实例。
  • 私有静态实例:单例类通常会维护一个私有的静态实例变量,用于保存唯一的实例。
  • 静态获取方法:单例类通常提供一个静态的获取方法,允许客户端代码访问单例实例。
  • 惰性初始化:单例实例通常是在首次访问时初始化的,以延迟实例化,提高性能。
  • 线程安全:单例模式通常需要考虑线程安全性,以防止多个线程同时创建实例。

二、适用场景

    单例模式适用于以下情况:

  1. 全局资源访问:当一个对象需要在系统中全局唯一访问时,如配置管理、日志记录、数据库连接等。
  2. 控制资源访问:当需要限制某种资源(如线程池、连接池、缓存)的访问数量时。
  3. 惰性初始化:当对象的创建和初始化可能具有较高的开销,但只在需要时才进行时。
  4. 线程池:用于创建线程池,以便在整个应用程序中维护一个可管理的线程池实例。

三、代码实战

饿汉式单例模式

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

原理:在类加载时,即实例化前,实例变量就已经被分配内存,因此保证了线程安全。但可能会浪费内存,因为不一定会使用该实例。

懒汉式单例模式

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

原理:该实现使用惰性初始化,只有在第一次调用 getInstance() 时才创建实例。通过加锁来确保线程安全,但可能引发性能问题。

双重检查锁定单例模式

public class DoubleCheckedSingleton {private volatile static DoubleCheckedSingleton instance;private DoubleCheckedSingleton() {// 私有构造函数}public static DoubleCheckedSingleton getInstance() {if (instance == null) {synchronized (DoubleCheckedSingleton.class) {if (instance == null) {instance = new DoubleCheckedSingleton();}}}return instance;}
}

原理:通过双重检查,确保只在第一次创建实例时使用同步锁,提高了性能。

静态内部类单例模式

public class InnerStaticSingleton {private InnerStaticSingleton() {// 私有构造函数}private static class SingletonHolder {private static final InnerStaticSingleton instance = new InnerStaticSingleton();}public static InnerStaticSingleton getInstance() {return SingletonHolder.instance;}
}

原理:利用Java类加载机制,保证只有在首次访问 getInstance() 时才加载内部类并创建实例。

四、实际应用举例

Java开发工具包(JDK)中有一些类和模块使用了单例模式来管理资源或确保全局唯一性。以下是一些示例:

  1. Runtime类:java.lang.Runtime类是Java的运行时系统入口,它使用了饿汉式单例模式来确保在应用程序中只有一个运行时实例。你可以通过Runtime.getRuntime()方法获取运行时对象的唯一实例,从而执行与运行时环境相关的操作。
  2. Calendar类:java.util.Calendar类是用于日期和时间操作的抽象类,它提供了一个getInstance()方法,返回特定日历系统的单例实例。这是一个懒汉式单例模式的示例,以确保只有一个日历实例。
  3. Desktop类:java.awt.Desktop类用于与桌面操作系统进行交互,如打开文件、浏览网页等。它使用了饿汉式单例模式来保证只有一个桌面对象的实例。

这些示例中的单例模式确保了全局唯一性,并用于管理特定资源或提供全局访问点。虽然不是所有JDK中的类都使用单例模式,但一些核心类确实使用了这种设计模式来确保一些特定的实例只存在一个。

Runtime解析

在 Java 的 Runtime 类中,使用了饿汉式单例模式来确保在应用程序中只有一个运行时实例。以下是 Runtime 类的部分代码,展示了如何使用单例模式来创建和返回运行时实例:

public class Runtime {private static Runtime currentRuntime = new Runtime();// 私有构造函数,防止外部实例化private Runtime() {}// 获取当前运行时实例public static Runtime getRuntime() {return currentRuntime;}// ...其他方法和功能...
}

解释:

  1. currentRuntime 是一个静态私有成员变量,它在类加载时被初始化。这就是饿汉式单例模式的特点,实例在类加载时就被创建,而不是在首次访问时才创建。
  2. 构造函数 private Runtime() 被声明为私有,这意味着外部代码无法直接实例化 Runtime 类。
  3. getRuntime() 方法是一个静态方法,用于获取当前运行时实例。它返回的是 currentRuntime 静态实例变量,确保每次调用 getRuntime() 方法都返回同一个实例。

这种设计确保了在整个应用程序中只有一个运行时实例,因为 currentRuntime 在类加载时就被创建,并且构造函数是私有的,外部代码无法创建新的实例。这是一种有效的方式,以确保只有一个全局运行时环境,以便执行与运行时环境相关的操作,如执行外部进程、获取系统信息等。

五、结论

以下是单例模式的一些总结和结论:
优点:

  1. 全局唯一性:单例模式确保一个类只有一个实例,因此能够全局唯一地管理特定类型的资源或状态。
  2. 惰性初始化:单例实例通常是在首次访问时初始化的,从而避免了不必要的资源浪费。
  3. 全局访问点:单例提供了一个全局访问点,允许应用程序中的其他对象轻松访问单例实例。
  4. 线程安全性:合理实现的单例模式可以确保多线程环境下的线程安全性,防止多次实例化。

缺点和注意事项:

  1. 过度使用:过度使用单例模式可能导致全局状态的滥用,使代码难以维护和测试。应谨慎使用。
  2. 不适用于所有情况:并非所有类都适合作为单例,有些类需要多个实例来独立处理不同任务。
  3. 测试困难:单例模式的全局状态共享可能会使单元测试变得复杂,因此需要额外的努力来测试应用。
  4. 多例模式:在某些情况下,可能需要多个实例,而不仅仅是一个全局唯一实例。

总的来说,单例模式是一种非常有用的设计模式,用于确保一个类只有一个实例。它在管理全局资源、限制实例数量以及提供全局访问点等方面发挥了重要作用。然而,在使用单例模式时,应慎重考虑,确保它真正符合应用程序的需求。此外,要注意线程安全性和测试问题,以确保单例实现的正确性。

点赞收藏,富婆包养✋✋

相关文章:

设计模式-单例模式实战

目录 一、引言二、适用场景三、代码实战饿汉式单例模式懒汉式单例模式双重检查锁定单例模式静态内部类单例模式 四、实际应用举例Runtime解析 五、结论 一、引言 单例模式是一种创建型设计模式,用于确保一个类只有一个实例,且提供全局访问点以访问该实例…...

requests库出现AttributeError问题的修复与替代方法

在使用App Engine时,开发者们通常会面临需要发送爬虫ip请求的情况,而Python中的requests库是一个常用的工具,用于处理爬虫ip请求。然而,在某些情况下,开发者可能会遇到一个名为AttributeError的问题,特别是…...

opencv-2D直方图

cv2.calcHist() 是 OpenCV 中用于计算直方图的函数。它可以计算一维或多维直方图,用于分析图像中像素值的分布。 基本的语法如下: hist cv2.calcHist(images, channels, mask, histSize, ranges[, hist[, accumulate]])参数说明: images:…...

读像火箭科学家一样思考笔记06_初学者之心

1. 专业化是目前流行的趋势 1.1. 通才(generalist)是指博而不精之人 1.2. 懂得的手艺越多,反而会家徒四壁 1.2.1. 希腊谚语 1.3. 这种态度代价很大,它阻断了不同学科思想的交融 2. 组合游戏 2.1. 某个行业的变革可能始于另一…...

中职组网络安全 Server-Hun-1.img Server-Hun-2.img

一串密码 smbuser用户和密码登录ssh还是失败提示需要密钥,尝试ftp登录成功 发现密钥存放在.ssh/下,在kali上生成一个密钥,通过上传到.ssh/下,将其替换掉 使用kali生成密钥 登录成功,但是无法拿到root目录下的flag 获取root用户权限…...

基于区域划分的GaN HEMT 准物理大信号模型

GaN HEMT器件的大信号等效电路模型分为经验基模型和物理基模型。经验基模型具有较高精度但参数提取困难,特别在GaN HEMT器件工艺不稳定的情况下不易应用。相比之下,物理基模型从器件工作机理出发,参数提取相对方便,且更容易更新和…...

laravel引入element-ui后,blade模板中使用elementui时,事件未生效问题(下载element-ui到本地直接引入项目)

背景 重构公司后台项目,使用了dcat-admin,但是dcat-admin有些前端功能不能满足需求。因此引入element-ui进行相关界面的优化 具体流程 1.下载element-ui到本地 2.进入如下目录 打开 node_modules\element-ui\lib 复制index.js 打开 node_modules/ele…...

【计算机网络笔记】路由算法之层次路由

系列文章目录 什么是计算机网络? 什么是网络协议? 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能(1)——速率、带宽、延迟 计算机网络性能(2)…...

【华为OD机试python】分糖果【2023 B卷|100分】

【华为OD机试】-真题 !!点这里!! 【华为OD机试】真题考点分类 !!点这里 !! 题目描述 小明从糖果盒中随意抓一把糖果,每次小明会取出一半的糖果分给同学们。 当糖果不能平均分配时,小明可以选择从糖果盒中(假设盒中糖果足够) 取出一个糖果或放回一个糖果。 小明最少需要多…...

ARM 汇编基础

我们在学习 STM32 的时候几乎没有用到过汇编,可能在学习 UCOS 、 FreeRTOS 等 RTOS 类操作系统移植的时候可能会接触到一点汇编。但是我们在进行嵌入式 Linux 开发的时候是绝 对要掌握基本的 ARM 汇编,因为 Cortex-A 芯片一上电 SP 指针还…...

虹科Pico汽车示波器 | 汽车免拆检修 | 2017款东风本田XR-V车转向助力左右不一致

一、故障现象 一辆2017款东风本田XR-V车,搭载R18ZA发动机,累计行驶里程约为4万km。车主反映,车辆行驶或静止时,向右侧转向比向左侧转向沉重。 二、故障诊断 接车后试车,起动发动机,组合仪表上无故障灯点亮&…...

阿里云服务器ECS经济型e实例优惠99元性能怎么样?

阿里云服务器ECS经济型e实例优惠99元性能怎么样?阿里云服务器优惠99元一年,配置为云服务器ECS经济型e实例,2核2G配置、3M固定带宽和40G ESSD Entry系统盘,CPU采用Intel Xeon Platinum架构处理器,2.5 GHz主频&#xff0…...

vue3引入vuex基础

一:前言 使用 vuex 可以方便我们对数据的统一化管理,便于各组件间数据的传递,定义一个全局对象,在多组件之间进行维护更新。因此,vuex 是在项目开发中很重要的一个部分。接下来让我们一起来看看如何使用 vuex 吧&#…...

C++二维数组中的查找

4. 二维数组中的查找 题目链接 牛客网 题目描述 给定一个二维数组,其每一行从左到右递增排序,从上到下也是递增排序。给定一个数,判断这个数是否在该二维数组中。 Consider the following matrix: [[1, 4, 7, 11, 15],[2, 5, 8, 12, 19],[3, 6, 9, 16, 22],[1…...

【计算思维】蓝桥杯STEMA 科技素养考试真题及解析 2

1、兰兰有一些数字卡片,从 1 到 100 的数字都有,她拿出几张数字卡片按照一定顺序摆放。想一想,第 5 张卡片应该是 A、11 B、12 C、13 D、14 答案:C 2、按照下图的规律,阴影部分应该填 A、 B、 C、 D、 答案&am…...

Qt+sqlite3使用事务提升插入效率

参考&#xff1a; 【精选】SQLite批量插入效率_sqlite 批量插入_PengX_Seek的博客-CSDN博客 (1)不使用事务时&#xff1a; clock_t t_start clock();QSqlQuery query(db);QString sql("insert into test(col1,col2) values(1,2);");for (int i 0; i < 1000; i…...

【深度学习】不用Conda在PP飞桨Al Studio三个步骤安装永久PyTorch环境

在 PaddlePaddle AI Studio 中使用 Python 虚拟环境安装 PyTorch 免责声明 在阅读和实践本文提供的内容之前&#xff0c;请注意以下免责声明&#xff1a; 侵权问题: 本文提供的信息仅供学习参考&#xff0c;不用做任何商业用途&#xff0c;如造成侵权&#xff0c;请私信我&am…...

SpringBoot:kaptcha生成验证码

GitHub项目地址&#xff1a;GitHub - penggle/kaptcha: kaptcha - A kaptcha generation engine. kaptcha介绍 kaptcha官网&#xff08;Google Code Archive - Long-term storage for Google Code Project Hosting.&#xff09;对其介绍如下&#xff0c; kaptcha十分易于安装…...

C/C++ 使用API实现数据压缩与解压缩

在Windows编程中&#xff0c;经常会遇到需要对数据进行压缩和解压缩的情况&#xff0c;数据压缩是一种常见的优化手段&#xff0c;能够减小数据的存储空间并提高传输效率。Windows提供了这些API函数&#xff0c;本文将深入探讨使用Windows API进行数据压缩与解压缩的过程&#…...

Visual Studio连接unity编辑器_unity基础开发教程

Visual Studio连接unity编辑器 问题描述解决方法意外情况 问题描述 当我们在unity编辑器中打开C#脚本的时候发现Visual Studio没有连接unity编辑器&#xff0c;在编写代码的时候也没有unity关键字的提醒。 简单来说就是敲代码没有代码提示。 解决方法 这时候需要在unity中进行…...

反射获取方法和属性

Java反射获取方法 在Java中&#xff0c;反射&#xff08;Reflection&#xff09;是一种强大的机制&#xff0c;允许程序在运行时访问和操作类的内部属性和方法。通过反射&#xff0c;可以动态地创建对象、调用方法、改变属性值&#xff0c;这在很多Java框架中如Spring和Hiberna…...

C++中string流知识详解和示例

一、概览与类体系 C 提供三种基于内存字符串的流&#xff0c;定义在 <sstream> 中&#xff1a; std::istringstream&#xff1a;输入流&#xff0c;从已有字符串中读取并解析。std::ostringstream&#xff1a;输出流&#xff0c;向内部缓冲区写入内容&#xff0c;最终取…...

IT供电系统绝缘监测及故障定位解决方案

随着新能源的快速发展&#xff0c;光伏电站、储能系统及充电设备已广泛应用于现代能源网络。在光伏领域&#xff0c;IT供电系统凭借其持续供电性好、安全性高等优势成为光伏首选&#xff0c;但在长期运行中&#xff0c;例如老化、潮湿、隐裂、机械损伤等问题会影响光伏板绝缘层…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程

本文较长&#xff0c;建议点赞收藏&#xff0c;以免遗失。更多AI大模型应用开发学习视频及资料&#xff0c;尽在聚客AI学院。 本文全面剖析RNN核心原理&#xff0c;深入讲解梯度消失/爆炸问题&#xff0c;并通过LSTM/GRU结构实现解决方案&#xff0c;提供时间序列预测和文本生成…...

Android第十三次面试总结(四大 组件基础)

Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成&#xff0c;用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机&#xff1a; ​onCreate()​​ ​调用时机​&#xff1a;Activity 首次创建时调用。​…...

使用Matplotlib创建炫酷的3D散点图:数据可视化的新维度

文章目录 基础实现代码代码解析进阶技巧1. 自定义点的大小和颜色2. 添加图例和样式美化3. 真实数据应用示例实用技巧与注意事项完整示例(带样式)应用场景在数据科学和可视化领域,三维图形能为我们提供更丰富的数据洞察。本文将手把手教你如何使用Python的Matplotlib库创建引…...

Scrapy-Redis分布式爬虫架构的可扩展性与容错性增强:基于微服务与容器化的解决方案

在大数据时代&#xff0c;海量数据的采集与处理成为企业和研究机构获取信息的关键环节。Scrapy-Redis作为一种经典的分布式爬虫架构&#xff0c;在处理大规模数据抓取任务时展现出强大的能力。然而&#xff0c;随着业务规模的不断扩大和数据抓取需求的日益复杂&#xff0c;传统…...

Elastic 获得 AWS 教育 ISV 合作伙伴资质,进一步增强教育解决方案产品组合

作者&#xff1a;来自 Elastic Udayasimha Theepireddy (Uday), Brian Bergholm, Marianna Jonsdottir 通过搜索 AI 和云创新推动教育领域的数字化转型。 我们非常高兴地宣布&#xff0c;Elastic 已获得 AWS 教育 ISV 合作伙伴资质。这一重要认证表明&#xff0c;Elastic 作为 …...

上位机开发过程中的设计模式体会(1):工厂方法模式、单例模式和生成器模式

简介 在我的 QT/C 开发工作中&#xff0c;合理运用设计模式极大地提高了代码的可维护性和可扩展性。本文将分享我在实际项目中应用的三种创造型模式&#xff1a;工厂方法模式、单例模式和生成器模式。 1. 工厂模式 (Factory Pattern) 应用场景 在我的 QT 项目中曾经有一个需…...

React从基础入门到高级实战:React 实战项目 - 项目五:微前端与模块化架构

React 实战项目&#xff1a;微前端与模块化架构 欢迎来到 React 开发教程专栏 的第 30 篇&#xff01;在前 29 篇文章中&#xff0c;我们从 React 的基础概念逐步深入到高级技巧&#xff0c;涵盖了组件设计、状态管理、路由配置、性能优化和企业级应用等核心内容。这一次&…...