设计模式-单例模式实战
目录
- 一、引言
- 二、适用场景
- 三、代码实战
- 饿汉式单例模式
- 懒汉式单例模式
- 双重检查锁定单例模式
- 静态内部类单例模式
- 四、实际应用举例
- Runtime解析
- 五、结论
一、引言
单例模式是一种创建型设计模式,用于确保一个类只有一个实例,且提供全局访问点以访问该实例。以下是单例模式的关键特点:
- 私有构造函数:单例类通常会将其构造函数声明为私有,以防止外部直接创建实例。
- 私有静态实例:单例类通常会维护一个私有的静态实例变量,用于保存唯一的实例。
- 静态获取方法:单例类通常提供一个静态的获取方法,允许客户端代码访问单例实例。
- 惰性初始化:单例实例通常是在首次访问时初始化的,以延迟实例化,提高性能。
- 线程安全:单例模式通常需要考虑线程安全性,以防止多个线程同时创建实例。
二、适用场景
单例模式适用于以下情况:
- 全局资源访问:当一个对象需要在系统中全局唯一访问时,如配置管理、日志记录、数据库连接等。
- 控制资源访问:当需要限制某种资源(如线程池、连接池、缓存)的访问数量时。
- 惰性初始化:当对象的创建和初始化可能具有较高的开销,但只在需要时才进行时。
- 线程池:用于创建线程池,以便在整个应用程序中维护一个可管理的线程池实例。
三、代码实战
饿汉式单例模式
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)中有一些类和模块使用了单例模式来管理资源或确保全局唯一性。以下是一些示例:
- Runtime类:java.lang.Runtime类是Java的运行时系统入口,它使用了饿汉式单例模式来确保在应用程序中只有一个运行时实例。你可以通过Runtime.getRuntime()方法获取运行时对象的唯一实例,从而执行与运行时环境相关的操作。
- Calendar类:java.util.Calendar类是用于日期和时间操作的抽象类,它提供了一个getInstance()方法,返回特定日历系统的单例实例。这是一个懒汉式单例模式的示例,以确保只有一个日历实例。
- 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;}// ...其他方法和功能... }解释:
- currentRuntime 是一个静态私有成员变量,它在类加载时被初始化。这就是饿汉式单例模式的特点,实例在类加载时就被创建,而不是在首次访问时才创建。
- 构造函数 private Runtime() 被声明为私有,这意味着外部代码无法直接实例化 Runtime 类。
- getRuntime() 方法是一个静态方法,用于获取当前运行时实例。它返回的是 currentRuntime 静态实例变量,确保每次调用 getRuntime() 方法都返回同一个实例。
这种设计确保了在整个应用程序中只有一个运行时实例,因为 currentRuntime 在类加载时就被创建,并且构造函数是私有的,外部代码无法创建新的实例。这是一种有效的方式,以确保只有一个全局运行时环境,以便执行与运行时环境相关的操作,如执行外部进程、获取系统信息等。
五、结论
以下是单例模式的一些总结和结论:
优点:
- 全局唯一性:单例模式确保一个类只有一个实例,因此能够全局唯一地管理特定类型的资源或状态。
- 惰性初始化:单例实例通常是在首次访问时初始化的,从而避免了不必要的资源浪费。
- 全局访问点:单例提供了一个全局访问点,允许应用程序中的其他对象轻松访问单例实例。
- 线程安全性:合理实现的单例模式可以确保多线程环境下的线程安全性,防止多次实例化。
缺点和注意事项:
- 过度使用:过度使用单例模式可能导致全局状态的滥用,使代码难以维护和测试。应谨慎使用。
- 不适用于所有情况:并非所有类都适合作为单例,有些类需要多个实例来独立处理不同任务。
- 测试困难:单例模式的全局状态共享可能会使单元测试变得复杂,因此需要额外的努力来测试应用。
- 多例模式:在某些情况下,可能需要多个实例,而不仅仅是一个全局唯一实例。
总的来说,单例模式是一种非常有用的设计模式,用于确保一个类只有一个实例。它在管理全局资源、限制实例数量以及提供全局访问点等方面发挥了重要作用。然而,在使用单例模式时,应慎重考虑,确保它真正符合应用程序的需求。此外,要注意线程安全性和测试问题,以确保单例实现的正确性。
点赞收藏,富婆包养✋✋
相关文章:
设计模式-单例模式实战
目录 一、引言二、适用场景三、代码实战饿汉式单例模式懒汉式单例模式双重检查锁定单例模式静态内部类单例模式 四、实际应用举例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主频࿰…...
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使用事务提升插入效率
参考: 【精选】SQLite批量插入效率_sqlite 批量插入_PengX_Seek的博客-CSDN博客 (1)不使用事务时: 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 免责声明 在阅读和实践本文提供的内容之前,请注意以下免责声明: 侵权问题: 本文提供的信息仅供学习参考,不用做任何商业用途,如造成侵权,请私信我&am…...
SpringBoot:kaptcha生成验证码
GitHub项目地址:GitHub - penggle/kaptcha: kaptcha - A kaptcha generation engine. kaptcha介绍 kaptcha官网(Google Code Archive - Long-term storage for Google Code Project Hosting.)对其介绍如下, kaptcha十分易于安装…...
C/C++ 使用API实现数据压缩与解压缩
在Windows编程中,经常会遇到需要对数据进行压缩和解压缩的情况,数据压缩是一种常见的优化手段,能够减小数据的存储空间并提高传输效率。Windows提供了这些API函数,本文将深入探讨使用Windows API进行数据压缩与解压缩的过程&#…...
Visual Studio连接unity编辑器_unity基础开发教程
Visual Studio连接unity编辑器 问题描述解决方法意外情况 问题描述 当我们在unity编辑器中打开C#脚本的时候发现Visual Studio没有连接unity编辑器,在编写代码的时候也没有unity关键字的提醒。 简单来说就是敲代码没有代码提示。 解决方法 这时候需要在unity中进行…...
AI推理优化工程2026:从模型压缩到推理加速的完整实战指南
引言:推理成本的现实困境 大模型的训练成本是一次性的,但推理成本是持续的。一家中型企业每天调用 GPT-4 级别模型处理 100 万次请求,月均 API 费用可能高达数十万元。更糟糕的是,许多企业在私有化部署时,GPU 的利用率…...
大语言模型微调实战:从LoRA原理到ChatGPT定制化应用
1. 项目概述:从原理到代码,深入理解ChatGPT的微调最近在GitHub上看到一个名为“ChatGPT_principle_fine-tuning_code_paper”的项目,它吸引我的地方在于,它试图将大语言模型(LLM)的核心原理、微调ÿ…...
5V转3.3V,用LDO还是DC-DC?实测对比纹波、功耗和成本,给你一个明确的答案
5V转3.3V电源方案深度评测:LDO与DC-DC的工程实践选择 在嵌入式系统设计中,电源转换电路的选择往往决定了产品的稳定性和能效表现。当面对5V到3.3V这一常见电压转换需求时,工程师们通常会在LDO(低压差线性稳压器)和DC-D…...
JSON Schema表单构建器:声明式配置驱动Web表单开发
1. 项目概述:一个开箱即用的表单构建器 如果你做过Web开发,尤其是后台管理系统,那你一定对表单深恶痛绝。重复的HTML结构、繁琐的验证逻辑、千篇一律的样式调整,还有那永远也填不完的字段映射和数据提交。每次接到一个“简单”的增…...
有效的括号
1.栈的经典应用,建议先去了解栈的基础题目链接:https://leetcode.cn/problems/valid-parentheses/视频讲解:https://www.bilibili.com/video/BV1AF411w78g2.代码class Solution { public:bool isValid(string s) {unordered_map<char, ch…...
npm install卡在git clone?别急着换镜像,先试试这个DNS刷新命令
npm install卡在git clone?别急着换镜像,先试试这个DNS刷新命令 作为一名前端开发者,相信大家都遇到过npm install卡在git clone阶段的尴尬情况。控制台不断输出Failed to connect to github.com port 443的错误信息,让人既焦虑又…...
ALLPCB 1美元6层PCB打样服务全解析
1. ALLPCB 1美元PCB打样服务深度体验报告作为一名硬件工程师,我深知PCB打样在项目开发中的重要性。最近发现ALLPCB推出的1美元6层板打样活动,第一时间下单体验并记录全过程,希望能为同行提供参考。ALLPCB成立于2015年,总部位于杭州…...
独立开发记录:我怎么把一个专注计时器做成了「声音护照」— iOS端技术拆解
为什么要做这个App 去年我给自己定了个规矩,每天至少专注写稿25分钟。试了一圈市面上的番茄钟,发现都是同一个套路:倒计时、叮一声、结束。 用了两周就不想打开了。没有留存感,没有成就感,就是个闹钟。 我想要的是——…...
Swift代码风格自动化检查终极指南:从混乱到规范的最佳实践
Swift代码风格自动化检查终极指南:从混乱到规范的最佳实践 【免费下载链接】swift-style-guide The official Swift style guide for Kodeco. 项目地址: https://gitcode.com/gh_mirrors/sw/swift-style-guide Swift代码风格自动化检查是提升开发效率和代码质…...
别再用Python写AI后端了!PHP 9.0原生协程实现类ChatGPT实时流响应——附可运行GitHub仓库(限前200名领取)
更多请点击: https://intelliparadigm.com 第一章:PHP 9.0异步编程与AI聊天机器人教程概览 PHP 9.0 引入了原生协程(Native Coroutines)与 async/await 语法支持,标志着 PHP 正式迈入现代异步编程时代。该版本底层基于…...
