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

Java学习36-Java 多线程安全:懒汉式和饿汉式

JAVA种有两种保证线程安全的方式,分别叫懒汉式Lazy Initialization和饿汉式Eager Initialization,以下是他们的区别:

  • 线程安全性:

懒汉式本身是非线程安全的,因为多个线程可能同时检查实例是否为null,并尝试同时创建实例,会导致出现多个实例。为了解决这个问题,需要额外的同步机制,如双重检查锁定(double-checked locking)或静态内部类等方式。
而饿汉式最开始就static和final了天生就是线程安全的。

  • 实例创建时机不同:

懒汉式在类被创建时不立即创建实例,而是在第一次调用 类名.getInstance() 方法时才创建实例,实现了延迟加载(非线程安全)。
饿汉式直接在最开始就static final SingletonEager instance = new SingletonEager()直接创建完毕了(自带安全属性)。

  • 资源加载和性能不同:

懒汉式(慢):延迟了实例的创建,只有在真正需要使用时才会进行初始化,因此可以节省资源。但在第一次调用 getInstance() 方法时,由于需要创建实例,可能会有一定的性能延迟。
饿汉式(快,浪费内存):对象在加载时已经创建,因此无论是否适用单例对象,都会占用一定内存。但是由于对象已经提前初始化,第一次调用getInstance方法速度会更快。

饿汉式(本身就是线程安全的)

饿汉式(Eager Initialization)开始就直接创建,不调用也存在,占内存,调用跑起来快,自带线程安全属性。

饿汉式举例:

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

使用举例:

package ThreadPool;public class Test {public static void main(String[] args) {EagerSingleton e1 = EagerSingleton.getInstance();EagerSingleton e2 = EagerSingleton.getInstance();System.out.println(e1==e2);}}class EagerSingleton{private EagerSingleton() {}private static final EagerSingleton instance = new EagerSingleton();public static EagerSingleton getInstance(){return instance;}
}

运行结果

true

懒汉式

懒汉式(Lazy Initialization)

下面是一个非线程安全的一般懒汉式示例(不建议使用,除非有额外的同步机制):

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

线程安全的懒汉式示例(使用双重检查锁定):

第一次判断if (instance == null) 再进行下面的线程synchronized, 如果实例已经存在,直接都不用管线程synchronized那些程序块,直接return输出了。

public class ThreadSafeLazySingleton {  private static volatile ThreadSafeLazySingleton instance;  private ThreadSafeLazySingleton() {  // 私有构造函数  }  public static ThreadSafeLazySingleton getInstance() {  if (instance == null) { // 第一次检查实例是否存在  synchronized (ThreadSafeLazySingleton.class) {  if (instance == null) { // 第二次检查实例是否存在  instance = new ThreadSafeLazySingleton();  }  }  }  return instance;  }  
}

补充知识点

volatile是一个关键字,用于修饰变量。当一个变量被声明为volatile时,它意味着这个变量在多线程环境下是可见的和有序的。这有助于确保线程安全,但它并不保证复合操作的原子性。例如,自增操作++实际上包括读取、增加和写入三个步骤,如果多个线程同时对一个volatile变量进行自增操作,那么结果可能会不正确。在下面的例子中volatile被用在在Bank instance的定义中。

示例2:构建一个银行单例,使用三个线程分别调用它,保证线程安全条件下(三个线程调用的是同一个银行instance),输出“线程名字+ My Private Bank is building up! ”

package ThreadPool;public class Test3 {public static void main(String[] args) {//创建三个BankThread对象BankThread b1 = new BankThread();BankThread b2 = new BankThread();BankThread b3 = new BankThread();//分别启动这三个线程,因为Bank类是单例的,因此所有线程都将获取到同一个Bank对象实例b1.start();b2.start();b3.start();}}//一个专门构建的可以调用Bank类的Thread类
class BankThread extends Thread{@Overridepublic void run() {Bank bank = Bank.getInstance();bank.PrintBank();}
}//构建Bank类,实现了懒汉单例模式
//两层if(instance == null)和 synchronized (Bank.class)确保线程安全
class Bank{private static volatile Bank instance;private Bank() {}static Bank getInstance(){if(instance == null){synchronized (Bank.class){if(instance == null){instance = new Bank();}}}return instance;}public void PrintBank(){System.out.println(Thread.currentThread().getName()+ " My Private Bank is building up! ");}
}

运行输出:


Thread-0 My Private Bank is building up! 
Thread-1 My Private Bank is building up! 
Thread-2 My Private Bank is building up! Process finished with exit code 0

饿汉式和懒汉式的主要区别在于实例的创建时机和线程安全性。饿汉式在类加载时即创建实例,线程安全且性能较高(首次调用速度快),但可能浪费资源(即使实例从未被使用)。懒汉式则延迟了实例的创建,节省了资源,但需要在多线程环境下采取额外的同步措施来保证线程安全。在实际应用中,应根据具体需求选择适合的实现方式。

相关文章:

Java学习36-Java 多线程安全:懒汉式和饿汉式

JAVA种有两种保证线程安全的方式,分别叫懒汉式Lazy Initialization和饿汉式Eager Initialization,以下是他们的区别: 线程安全性: 懒汉式本身是非线程安全的,因为多个线程可能同时检查实例是否为null,并尝…...

sql常用之CASE WHEN THEN

sql常用之CASE WHEN THEN SQL中的 CASE 类似编程语言里的 if-then-else 语句,用做逻辑判断。可以用于SELECT语句中,也可以用在WHERE,GROUP BY 和 ORDER BY 子句;可以单独使用,也可以和聚合函数结合使用。 语法&#…...

【PduR路由】IPduM模块详细介绍

目录 1.IpduM功能简介 2.IpduM模块依赖的其他模块 2.1RTE (BSW Scheduler) 2.2PDU Router 2.3COM 3.IpduM功能详解 3.1 功能概述 3.2 I-PDU多路复用I-PDU Multiplexing 3.2.1 Definitions and Layout 3.2.2通用功能描述 General 3.2.3模块初始化 Initialization 3.…...

【MySQL】6.MySQL主从复制和读写分离

主从复制 主从复制与读写分离 通常数据库的读/写都在同一个数据库服务器中进行; 但这样在安全性、高可用性和高并发等各个方面无法满足生产环境的实际需求; 因此,通过主从复制的方式同步数据,再通过读写分离提升数据库的并发负载…...

Lucene及概念介绍

Lucene及概念介绍 基础概念倒排索引索引合并分析查询语句的构成 基础概念 Document:我们一次查询或更新的载体,对比于实体类 Field:字段,是key-value格式的数据,对比实体类的字段 Item:一个单词&#xff0…...

密码算法概论

基本概念 什么是密码学? 简单来说,密码学就是研究编制密码和破译密码的技术科学 例题: 密码学的三个阶段 古代到1949年:具有艺术性的科学1949到1975年:IBM制定了加密标准DES1976至今:1976年开创了公钥密…...

实时数仓之实时数仓架构(Hudi)

目前比较流行的实时数仓架构有两类,其中一类是以FlinkDoris为核心的实时数仓架构方案;另一类是以湖仓一体架构为核心的实时数仓架构方案。本文针对FlinkHudi湖仓一体架构进行介绍,这套架构的特点是可以基于一套数据完全实现Lambda架构。实时数…...

2022-04-15_for循环等_作业

for循环 编写程序数一下 1到 100 的所有整数中出现多少个数字9计算1/1-1/21/3-1/41/5 …… 1/99 - 1/100 的值&#xff0c;打印出结果求10 个整数中最大值在屏幕上输出9*9乘法口诀表二分查找 编写程序数一下 1到 100 的所有整数中出现多少个数字9 #include <stdio.h>in…...

脑机辅助推导算法

目录 一&#xff0c;背景 二&#xff0c;华容道中道 1&#xff0c;问题 2&#xff0c;告诉脑机如何编码一个正方形格子 3&#xff0c;让脑机汇总信息 4&#xff0c;观察图&#xff0c;得到启发式算法 5&#xff0c;根据启发式算法求出具体解 6&#xff0c;可视化 一&am…...

【原创教程】三菱FX PLC控制FR-E740变频器

变频器的使用 1. 使用三菱FX PLC 控制变频器时,接线图请按下图所示接线。 各个端子的说明如下: R、S、T:变频器电源,E740变频器电源位3相380V。 STF:正转启动, STF信号ON时为正转、OFF时为停止指令。 STR :反转启动,STR信号ON时为反转、OFF时为停止指令。 RH、RM、RL…...

重读Java设计模式: 深入探讨建造者模式,构建复杂对象的优雅解决方案

引言 在软件开发中&#xff0c;有时需要构建具有复杂结构的对象&#xff0c;如果直接使用构造函数或者 setter 方法逐个设置对象的属性&#xff0c;会导致代码变得冗长、难以维护&#xff0c;并且容易出错。为了解决这个问题&#xff0c;我们可以使用建造者模式。 一、建造者…...

C语言数据结构易错知识点(6)(快速排序、归并排序、计数排序)

快速排序属于交换排序&#xff0c;交换排序还有冒泡排序&#xff0c;这个太简单了&#xff0c;这里就不再讲解。 归并排序和快速排序都是采用分治法实现的排序&#xff0c;理解它们对分支思想的感悟会更深。 计数排序属于非比较排序&#xff0c;在数据集中的情况下可以考虑使…...

使用 React Router v6.22 进行导航

使用 React Router v6.22 进行导航 React Router v6.22 是 React 应用程序中最常用的路由库之一&#xff0c;提供了强大的导航功能。本文将介绍如何在 React 应用程序中使用 React Router v6.22 进行导航。 安装 React Router 首先&#xff0c;我们需要安装 React Router v6…...

单链表的插入和删除

一、插入操作 按位序插入&#xff08;带头结点&#xff09;&#xff1a; ListInsert(&L,i,e):插入操作。在表L中的第i个位置上插入指定元素e。 typedef struct LNode{ElemType data;struct LNode *next; }LNode,*LinkList;//在第i 个位置插插入元素e (带头结点) bool Li…...

全量知识系统 之“程序”详细设计 之 “絮”---开端“元素周期表”表示的一个“打地鼠”游戏

全量知识系统 之“程序”详细设计 概述-概要和纪要 序 絮&#xff08;一个极简的开场白--“全量知识系统”自我介绍&#xff09; 将整个“人生”的三个阶段 比作“幼稚园”三班 &#xff1a; 第一步【想】-- “感性”思维游戏&#xff1a;打地鼠 。学前教育-新生期&#x…...

【详细讲解WebView的使用与后退键处理】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…...

【Linux多线程】生产者消费者模型

【Linux多线程】生产者消费者模型 目录 【Linux多线程】生产者消费者模型生产者消费者模型为何要使用生产者消费者模型生产者消费者的三种关系生产者消费者模型优点基于BlockingQueue的生产者消费者模型C queue模拟阻塞队列的生产消费模型 伪唤醒情况&#xff08;多生产多消费的…...

Django屏蔽Server响应头信息

一、背景 最近我们被安全部门的漏洞扫描工具扫出了一个服务端口的漏洞。这个服务本身是一个Django启动的web服务&#xff0c;并且除了登录页面&#xff0c;其它页面或者接口都需要进行登录授权才能进行访问。 漏洞扫描信息和提示修复信息如下: 自然这些漏洞如何修复&#xff0c…...

前端对数据进行分组和计数处理

js对数组数据的处理&#xff0c;添加属性&#xff0c;合并表格数据。 let data[{id:1,group_id:111},{id:2,group_id:111},{id:3,group_id:111},{id:4,group_id:222},{id:5,group_id:222} ]let tempDatadata; tempDatatempData.reduce((arr,item)>{let findarr.find(i>i…...

synchronized 和 lock

synchronized 和 Lock 都是 Java 中用于实现线程同步的机制&#xff0c;它们都可以保证线程安全。 # synchronized 介绍与使用 synchronized 可用来修饰普通方法、静态方法和代码块&#xff0c;当一个线程访问一个被 synchronized 修饰的方法或者代码块时&#xff0c;会自动获…...

企业微信消息发送踩坑实录:.NET Core下处理AccessToken过期与消息安全的最佳实践

企业微信消息发送实战&#xff1a;.NET Core中的AccessToken管理与消息安全策略 当企业微信API集成到生产环境时&#xff0c;开发者常会遇到两个看似简单却暗藏玄机的问题&#xff1a;AccessToken突然失效导致消息发送失败&#xff0c;以及敏感信息传输时的安全风险。本文将分享…...

终极免费文档下载指南:如何用kill-doc脚本轻松获取百度文库、豆丁网等30+平台资源

终极免费文档下载指南&#xff1a;如何用kill-doc脚本轻松获取百度文库、豆丁网等30平台资源 【免费下载链接】kill-doc 看到经常有小伙伴们需要下载一些免费文档&#xff0c;但是相关网站浏览体验不好各种广告&#xff0c;各种登录验证&#xff0c;需要很多步骤才能下载文档&a…...

CipherChat:基于词元替换的端到端加密大模型对话方案解析

1. 项目概述&#xff1a;当大模型对话遇上密码学最近在折腾大语言模型&#xff08;LLM&#xff09;应用开发的朋友&#xff0c;可能都遇到过同一个头疼的问题&#xff1a;如何保证用户和模型之间对话的隐私和安全&#xff1f;我们辛辛苦苦搭建的智能客服、个人助理或者创意写作…...

DeepSeek模型服务Kubernetes化迁移 checklist(含CRD定义、ServiceMesh适配、TLS双向认证配置)

更多请点击&#xff1a; https://intelliparadigm.com 第一章&#xff1a;DeepSeek模型服务Kubernetes化迁移全景概览 将DeepSeek系列大语言模型&#xff08;如DeepSeek-V2、DeepSeek-Coder&#xff09;从单机或虚拟机部署迁移至Kubernetes集群&#xff0c;是支撑高并发推理、…...

3步构建你的第二大脑:Obsidian知识管理系统实战指南

3步构建你的第二大脑&#xff1a;Obsidian知识管理系统实战指南 【免费下载链接】obsidian-template Starter templates for Obsidian 项目地址: https://gitcode.com/gh_mirrors/ob/obsidian-template 你是否曾为笔记杂乱无章而烦恼&#xff1f;是否在需要某个知识点时…...

MagiskBoot:Android启动镜像解构与重构引擎深度解析

MagiskBoot&#xff1a;Android启动镜像解构与重构引擎深度解析 【免费下载链接】Magisk The Magic Mask for Android 项目地址: https://gitcode.com/GitHub_Trending/ma/Magisk MagiskBoot作为Magisk生态系统的核心组件&#xff0c;专门负责Android启动镜像的多格式解…...

AI编程助手效率革命:结构化配置与提示词工程实战

1. 项目概述&#xff1a;一个为AI编程时代量身定制的开发者工具箱如果你和我一样&#xff0c;日常开发已经离不开像 Cursor 和 Claude 这样的 AI 编程助手&#xff0c;那你肯定也遇到过类似的困扰&#xff1a;每次开启一个新项目&#xff0c;或者在不同项目间切换时&#xff0c…...

初次使用Taotoken平台从注册到完成API调用的全程指引

&#x1f680; 告别海外账号与网络限制&#xff01;稳定直连全球优质大模型&#xff0c;限时半价接入中。 &#x1f449; 点击领取海量免费额度 初次使用Taotoken平台从注册到完成API调用的全程指引 对于初次接触大模型API的开发者而言&#xff0c;从注册平台到成功发出第一个…...

Mega:基于上下文工程的Brainbase平台AI开发效率革命

1. 项目概述&#xff1a;Mega&#xff0c;你的Brainbase平台AI工程专家如果你正在使用Claude Code、Cursor或者任何能读取文件的AI编程工具来构建基于Brainbase平台的对话式AI应用&#xff0c;那么你很可能遇到过这样的困境&#xff1a;你需要花费大量时间向AI解释Brainbase的架…...

生成式AI破解基因型-表型关联:AIPheno项目实战解析

1. 项目概述&#xff1a;当生成式AI遇见基因表型 如果你在生物信息学或者遗传育种领域工作&#xff0c;最近几年一定被“基因型-表型关联”这个老大难问题折磨过。我们手里有海量的基因组测序数据&#xff08;基因型&#xff09;&#xff0c;也积累了大量的生物体性状数据&…...