单例模式懒汉式、饿汉式(线程安全)
- 饿汉式单线程安全吗
饿汉式单例(Eager Singleton)是线程安全的。这种实现方式在类加载时就创建了单例实例,因此在多线程环境中,不存在多个线程同时创建实例的问题。
饿汉式单例的实现
以下是一个饿汉式单例的示例:
public class EagerSingleton {// 在类加载时立即创建实例private static final EagerSingleton instance = new EagerSingleton();// 私有构造函数private EagerSingleton() {// Initialization code}// 提供全局访问点public static EagerSingleton getInstance() {return instance;}
}
线程安全的原因
-
提前实例化:通过在类加载时创建实例,确保了在任何线程访问
getInstance()方法之前,实例已经被创建。 -
无同步开销:由于实例在类加载时就已创建,后续的访问不需要任何同步操作,提高了性能。
适用场景
饿汉式单例适合以下场景:
- 需要确保一定会使用该实例,并且不怕在启动时占用内存。
- 在应用启动时就能确定实例的创建。
缺点
饿汉式单例的缺点在于:
- 可能导致资源浪费:如果实例在运行时并不一定会被使用,它仍然会在启动时占用内存。
- 不支持懒加载:无论是否需要,实例都会在类加载阶段创建。
总之,饿汉式单例是线程安全的,适合特定场景,但在设计时需要根据需求仔细选择实现方式。
- 懒汉式加载时线程安全的吗?
懒汉式单例(Lazy Singleton)在默认实现中并不是线程安全的。多个线程同时访问 getInstance() 方法时,可能会导致多个实例的创建。为了确保线程安全,需要采取适当的同步措施。
懒汉式单例的基本实现
以下是一个不安全的懒汉式单例示例:
public class LazySingleton {private static LazySingleton instance;private LazySingleton() {// Initialization code}public static LazySingleton getInstance() {if (instance == null) {instance = new LazySingleton(); // 可能会导致多个线程创建多个实例}return instance;}
}
线程不安全的原因
- 竞态条件:多个线程同时调用
getInstance()方法时,可能会同时发现instance为null,从而各自创建新的实例,导致多个实例存在。
确保线程安全的方式
- 使用
synchronized关键字:
public class LazySingleton {private static LazySingleton instance;private LazySingleton() {// Initialization code}public static synchronized LazySingleton getInstance() {if (instance == null) {instance = new LazySingleton();}return instance;}
}
这种方式确保了在同一时刻只有一个线程可以执行 getInstance() 方法,避免了多个实例的创建。
- 双重检查锁定(Double-Checked Locking):
public class LazySingleton {private static volatile LazySingleton instance;private LazySingleton() {// Initialization code}public static LazySingleton getInstance() {if (instance == null) {synchronized (LazySingleton.class) {if (instance == null) {instance = new LazySingleton();}}}return instance;}
}
volatile 关键字的作用:
- 防止指令重排序:在没有 volatile 的情况下,JVM 可能会对对象的初始化过程进行优化,导致指令重排序。例如,在创建对象时,可能会先返回对象引用,然后再初始化对象。这将导致某些线程可能获取到尚未完全初始化的实例。
确保可见性:volatile 关键字确保了当一个线程修改 instance 时,其他线程能够立即看到这个变化。这避免了由于线程缓存导致的可见性问题。 - 优点:
性能优化:使用双重检查锁定可以避免每次调用 getInstance() 时都进行同步,只有在 instance 为 null 时才会加锁。这样在多次调用时,性能开销显著降低。
延迟初始化:实例仅在第一次调用时创建,避免了实例的早期创建,节省资源。
适用场景:
适合需要懒加载的单例模式,特别是在高并发的环境下,能够有效地减少锁的开销。
3.通过静态内部类实现,利用 Java 的类加载机制确保线程安全。
public class Singleton {private Singleton() {// Initialization code}private static class SingletonHelper {private static final Singleton INSTANCE = new Singleton();}public static Singleton getInstance() {return SingletonHelper.INSTANCE;}
}
使用静态内部类来实现单例模式是一种优雅且高效的方法。以下是这种实现方式的一些主要优点:
-
- 线程安全
类加载机制:静态内部类的实例是在第一次调用 getInstance() 方法时加载的,这意味着在类加载过程中不会创建实例,从而确保了线程安全。
避免同步开销:由于实例在静态内部类中创建,只有在需要时才会被加载,因此不需要在方法上添加同步锁,从而减少了性能开销。
- 线程安全
-
- 延迟初始化
按需创建:实例仅在第一次调用 getInstance() 时创建,这样可以避免在应用启动时就创建资源,节省了内存和其他资源的使用。
- 延迟初始化
-
- 避免反序列化问题
反序列化保护:如果实现了 Serializable 接口,静态内部类的单例实现可以防止通过反序列化创建多个实例。因为反序列化时,静态内部类的静态字段会被正确初始化,确保返回的仍然是同一个实例。
- 避免反序列化问题
-
- 简单易读
代码清晰:静态内部类的实现方式简洁明了,易于理解和维护。相较于其他实现方式(如双重检查锁定),代码量较少,逻辑更加直观。
- 简单易读
-
- 兼容性
Java 语言特性:这种实现方式利用了 Java 的类加载机制,是一种符合 Java 语言设计的优雅方案,能够很好地与其他 Java 特性结合使用。
- 兼容性
总结
默认的懒汉式单例实现是线程不安全的。要确保线程安全,可以使用同步机制或其他设计模式。推荐静态内部类来实现
以下是使用静态内部类实现的单例模式的示例,包括一个 main 函数,展示如何调用并验证单例的行为。
单例模式实现
public class Singleton {private Singleton() {// Initialization code}private static class SingletonHelper {private static final Singleton INSTANCE = new Singleton();}public static Singleton getInstance() {return SingletonHelper.INSTANCE;}
}
主函数示例
public class Main {public static void main(String[] args) {// 创建多个线程来测试单例Runnable task = () -> {Singleton instance = Singleton.getInstance();System.out.println("Instance HashCode: " + instance.hashCode());};// 启动多个线程Thread thread1 = new Thread(task);Thread thread2 = new Thread(task);Thread thread3 = new Thread(task);thread1.start();thread2.start();thread3.start();// 等待线程执行完成try {thread1.join();thread2.join();thread3.join();} catch (InterruptedException e) {e.printStackTrace();}// 主线程再次获取实例Singleton mainInstance = Singleton.getInstance();System.out.println("Main Thread Instance HashCode: " + mainInstance.hashCode());}
}
示例说明
-
单例类:
Singleton类使用静态内部类实现了单例模式,确保了线程安全和延迟初始化。 -
主函数:
- 定义了一个
Runnable任务,任务中调用Singleton.getInstance()并打印实例的哈希码。 - 启动了三个线程来并发访问单例实例。
- 主线程最后再次调用
getInstance()并打印该实例的哈希码。
- 定义了一个
运行结果
你应该会看到所有线程打印的哈希码相同,表明它们获取的是同一个实例。例如:
Instance HashCode: 123456789
Instance HashCode: 123456789
Instance HashCode: 123456789
Main Thread Instance HashCode: 123456789
总结
这个示例展示了如何使用静态内部类实现单例模式,并通过多线程验证了其线程安全性。所有线程和主线程都获取到了同一个实例,验证了单例模式的有效性。
相关文章:
单例模式懒汉式、饿汉式(线程安全)
饿汉式单线程安全吗 饿汉式单例(Eager Singleton)是线程安全的。这种实现方式在类加载时就创建了单例实例,因此在多线程环境中,不存在多个线程同时创建实例的问题。 饿汉式单例的实现 以下是一个饿汉式单例的示例: …...
Cursor登录按钮点击没反应
问题 系统:Windows11 Cursor:Cursor 0.44.9 当安装Cursor打开进行登录时,点击Sign in没反应 解决方案 1.打开window11的设置 2.点击应用中的默认应用 3.在设置应用程序的默认值中搜索Google(没有Google浏览器的尝试下载一个&a…...
论文实现:Reactive Nonholonomic Trajectory Generation via Parametric Optimal Control
1. 多项式螺旋 曲率: κ ( s ) a 0 a 1 s a 2 s 2 a 3 s 3 a 4 s 4 a 5 s 5 \begin{align} \kappa(s) a_0 a_1s a_2s^2 a_3s^3 a_4s^4 a_5s^5 \end{align} κ(s)a0a1sa2s2a3s3a4s4a5s5 机器人朝向: θ ( s ) a 0 s a 1 …...
基于单片机中药存放环境监测系统的实现
基于单片机中药存放环境监测系统的实现 项目开发背景 随着现代中药的广泛应用,中药材的存储环境对其质量有着至关重要的影响。温湿度、烟雾、火灾等环境因素,若不加以控制,将会导致中药材失效或变质。因此,设计一个基于单片机的…...
九垠赢+商业管理系统 Common.ashx 文件上传致RCE漏洞复现
0x01 产品简介 九垠赢+商业管理系统是基于互联网技术的进销存管理软件,适用于新零售背景下各种业态的线上线下一体化的商超经营管理。赢+ERP以商业管理系统为底座,融合了多种软、硬件解决方案,实现了从企业、供应商、三方平台到顾客等日常管理的全线数字化、智能化和移动化…...
速盾:服务器CDN加速解析的好处有哪些呢?
随着互联网应用的普及,越来越多的企业开始关注如何提升网站的访问速度和用户体验。为了实现这一目标,许多企业选择使用CDN(内容分发网络)来加速网站的内容分发。CDN通过在全球范围内分布多个节点,将内容缓存到离用户最…...
C++ 设计模式:备忘录模式(Memento Pattern)
链接:C 设计模式 链接:C 设计模式 - 状态模式 备忘录模式(Memento Pattern)是一种行为设计模式,它允许在不破坏封装性的前提下捕获和恢复对象的内部状态。这个模式在需要保存和恢复对象状态的场景中非常有用ÿ…...
Android 系统 Activity 系统层深度定制的方法、常见问题以及解决办法
Android 系统 Activity 系统层深度定制的方法、常见问题以及解决办法 目录 引言Activity 系统层概述Activity 系统架构图Activity 系统层深度定制的方法 4.1 自定义 Activity 生命周期4.2 自定义 Activity 启动流程4.3 自定义 Activity 转场动画4.4 自定义 Activity 窗口管理4…...
PDF怎么压缩得又小又清晰?5种PDF压缩方法
PDF 文件在日常办公与学习中使用极为频繁,可想要把它压缩得又小又清晰却困难重重。一方面,PDF 格式本身具有高度兼容性,集成了文字、图像、矢量图等多样元素,压缩时难以兼顾不同元素特性,稍不注意,文字就会…...
YK人工智能(三)——万字长文学会torch深度学习
2.1 张量 本节主要内容: 张量的简介PyTorch如何创建张量PyTorch中张量的操作PyTorch中张量的广播机制 2.1.1 简介 几何代数中定义的张量是基于向量和矩阵的推广,比如我们可以将标量视为零阶张量,矢量可以视为一阶张量,矩阵就是…...
关于CISP报名费用详情
CISP即“注册信息安全专业人员”,是中国信息安全测评中心实施的国家认证项目,旨在培养信息安全领域的专业人才。对于有意报考CISP的考生而言,了解报名考试费用是备考过程中不可或缺的一环。 CISP的报名考试费用主要包括培训费用、考试费用、…...
vim 按下esc后取消高亮
配置 ideavimrc文件 " 按下 Esc 键时自动取消高亮 " 在普通模式下按下 Esc 键取消高亮 nnoremap <Esc> :nohlsearch<CR> " 在插入模式下按下 Esc 键取消高亮 inoremap <Esc> <Esc>:nohlsearch<CR>" 额外:当退出…...
SwiftUI:多语言实现富文本插值
实现的UI需求: 要求: 英文显示:3068 people have joined this plan today! 中文显示:今日有 3068 人已加入此计划! 实现代码: Text(AttributedString(localized:"**\(payPeoples)** people have joi…...
操作系统基础
概念 控制和管理操作系统软件和硬件 用户观点:用户和计算机之间的接口 基本概念 特征 并发,共享,虚拟,异步 功能 处理机管理,存储器管理,文件管理,设备管理 作为用户和硬件接口 命令接口&am…...
函数调用流程可用工具
方法 1、gdb,这个网上很多找找就有 2、tcmalloc,直接在调用的地方调一个malloc就可以看到调用的流程,具体见tcmalloc 3、backtrace,个人感觉这么最方便,具体见backtrace...
UniApp 页面布局基础
一、UniApp 页面布局简介 在当今的移动应用开发领域,跨平台开发已成为一种主流趋势。UniApp作为一款极具影响力的跨平台开发框架,凭借其“一套代码,多端运行”的特性,为开发者们提供了极大的便利,显著提升了开发效率。…...
2D图像测量到3D点云之物体三维尺寸测量!!!!
0,引言 本文将从双目采集的2D图像到3D点云进行转化,并进行物体尺寸测量,旨在为读者展示2D图像如何关联3D点云,并进行相关工业应用。 将2D图像转化为3D点云,并进行物体尺寸测量的技术,在工业领域有着广泛的…...
[TOTP]android kotlin实现 totp身份验证器 类似Google身份验证器
背景:自己或者公司用一些谷歌身份验证器或者microsoft身份验证器,下载来源不明,或者有广告,使用不安全。于是自己写一个,安全放心使用。 代码已开源:shixiaotian/sxt-android-totp: android totp authenti…...
2025决战智驾:从中阶卷到L3,车企需要抓好一个数据闭环
作者 |王博 编辑 |德新 全国都能开之后,智驾继续走向哪里? 2024年末,大部分主流车企已经实现了无(高精度)图全国都能开。而第一梯队的玩家,从以规则为主的算法框架,向神经网络模型为主的新架构…...
电子电气架构 --- 汽车电子电器设计概述
我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 所谓鸡汤,要么蛊惑你认命,要么怂恿你拼命,但都是回避问题的根源,以现象替代逻辑,以情绪代替思考,把消极接受现实的懦弱,伪装成乐观面对不幸的…...
WRF和WPS模型在Ubuntu系统上的安装与常见问题解决指南
1. WRF和WPS模型简介 WRF(Weather Research and Forecasting)模型是一个广泛应用于气象研究和业务预报的中尺度数值天气预报系统。它由美国国家大气研究中心(NCAR)等多个机构联合开发,能够模拟从全球尺度到云尺度的各种…...
<最小生成树> 1349:【例4-10】最优布线问题
1349:【例4-10】最优布线问题时间限制: 1000 ms 内存限制: 65536 KB 提交数:12074 通过数: 7598【题目描述】学校有n台计算机,为了方便数据传输,现要将它们用数据线连接起来。两台计算机被连接是指它们有数据线连接。由于计算机所…...
Vue项目本地部署神器:phpStudy+nginx配置全流程(支持多项目同时运行)
Vue项目本地部署神器:phpStudynginx配置全流程(支持多项目同时运行) 每次在本地调试Vue项目时,你是否也遇到过这样的困扰?项目打包后需要部署到服务器才能预览效果,但配置nginx对前端开发者来说又太过复杂。…...
5分钟实现零代码GUI开发!ImStudio让Dear ImGui界面设计效率倍增
5分钟实现零代码GUI开发!ImStudio让Dear ImGui界面设计效率倍增 【免费下载链接】ImStudio GUI layout designer for Dear ImGui 项目地址: https://gitcode.com/gh_mirrors/im/ImStudio 作为开发者,你是否曾为编写GUI界面而头疼?是否…...
Smithbox终极指南:5个技巧让你轻松掌握魂系列游戏修改艺术
Smithbox终极指南:5个技巧让你轻松掌握魂系列游戏修改艺术 【免费下载链接】Smithbox Smithbox is a modding tool for Elden Ring, Armored Core VI, Sekiro, Dark Souls 3, Dark Souls 2, Dark Souls, Bloodborne and Demons Souls. 项目地址: https://gitcode.…...
暗黑3一键宏终极指南:D3keyHelper让你的游戏效率提升300%
暗黑3一键宏终极指南:D3keyHelper让你的游戏效率提升300% 【免费下载链接】D3keyHelper D3KeyHelper是一个有图形界面,可自定义配置的暗黑3鼠标宏工具。 项目地址: https://gitcode.com/gh_mirrors/d3/D3keyHelper 还在为暗黑3中重复的技能按键感…...
STM32F0实战:基于HAL库开发【4.6】
21.2 USB模块STM32F0072VBT6微控制器片内集成有符合USB2.0全速设备技术规范要求的USB模块,通过该模块可以实现与PC主机的USB通信连接,进一步拓展将该系列微控制器的应用范围。21.2.1 USB模块的结构STM32F072VBT6微控制器片内集成的USB模块,可…...
Mac Mouse Fix:让普通鼠标释放专业级生产力
Mac Mouse Fix:让普通鼠标释放专业级生产力 【免费下载链接】mac-mouse-fix Mac Mouse Fix - Make Your $10 Mouse Better Than an Apple Trackpad! 项目地址: https://gitcode.com/GitHub_Trending/ma/mac-mouse-fix 还在忍受MacOS下鼠标滚动卡顿、侧键功能…...
HY-SRF05超声波模块的5个常见误区及优化技巧(附STM32代码)
HY-SRF05超声波模块的5个常见误区及优化技巧(附STM32代码) 在嵌入式开发中,HY-SRF05超声波模块因其成本低廉、使用简单而广受欢迎。但很多开发者在使用过程中常会遇到测量不稳定、精度不足等问题。本文将深入剖析五个最常见的误区,…...
Zotero Reference学术文献管理工具全攻略:从入门到精通
Zotero Reference学术文献管理工具全攻略:从入门到精通 【免费下载链接】zotero-reference PDF references add-on for Zotero. 项目地址: https://gitcode.com/gh_mirrors/zo/zotero-reference 一、核心价值:重构学术文献管理流程 节省70%文献整…...
