【JavaEE】——单例模式引起的多线程安全问题:“饿汉/懒汉”模式,及解决思路和方法(面试高频)

阿华代码,不是逆风,就是我疯,你们的点赞收藏是我前进最大的动力!!希望本文内容能够帮助到你!

目录

一:单例模式(singleton)

1:概念

二:“饿汉模式”

1:前引

2:代码编译

3:代码分析

4:解释为什么叫“饿汉模式”

三:“懒汉”模式

1:前引

2:代码编译

3:代码分析

4:“懒汉”的优点

5:比较“饿汉”和“懒汉”

四:“饿汉”模式线程安全问题

五:“懒汉”模式的线程安全问题

1:重复创建实例

2:解决问题

(1)解决思路的核心本质:

(2)缺点:

①效率降低:

②不可预期:

3:优化效率问题

(1)问题解释

(2)解决方法

六:指令重排序问题

前引:

1:代码拆分三指令

2:分析

3:代码解释

4:解锁思路

(1)复习volatile两个功能


一:单例模式(singleton)

1:概念

单例模式就是,在java进程中,要求指定的类,只能有一个对象

我们通过一些特殊的技巧来确保,我们的实例(对象)只有一个——换句话说,就是如果我们尝试new多个实例,编译器就会报错。

二:“饿汉模式”

1:前引

(1)知识科普

我们先认识俩个单词——singleton(单例模式)和getInstance(获取实例)

(2)每个类只能有一个类对象,比如Thread类,Thread.class(获取到Thread这个类的类对象)它是从.class文件加载到内存当中的

2:代码编译

package thread;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: Hua YY
 * Date: 2024-09-24
 * Time: 13:20
 */
class Singleton{
    private static Singleton instance = new Singleton();
    public static Singleton getInstance(){//返回值类型为Singleton
        return instance;
    }
    private Singleton(){
        //静态构造方法让你访问不到
    }

}
public class ThreadDemon28 {
    //“饿汉模式”
    public static void main(String[] args) {
        Singleton s1 = Singleton.getInstance();
        Singleton s2 = Singleton.getInstance();
        System.out.println(s1 == s2);//判断s1和s2是否指向的是同一个对象

    }
}

3:代码分析

最后我们在main方法中比较s1,s2获取到的地址是否相同(s1,s2所指向的对象)

4:解释为什么叫“饿汉模式”

因为我们在类加载的时候就创建了这个实例,(初始化静态成员变量),这个时机非常的早——相当于程序一启动,实例就创建好了。所以就用“饿汉”来形容创建实例非常的迫切非常早

三:“懒汉”模式

1:前引

“懒汉”模式相对于“饿汉”模式不同的地方在于创建实例的时机更加晚一些,下面我们通过代码来进行展示

2:代码编译

package thread;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: Hua YY
 * Date: 2024-09-24
 * Time: 13:52
 */
class SingletonLazy{
    private static SingletonLazy instance = null;
    public static SingletonLazy getInstance(){
        if (instance == null){   //如果instance为null,则进入if创建实例,创建实例的时机是第一次调用getInstance这个方法
            instance = new SingletonLazy();
        }
        return instance;//第二次第三次调用getInstance()这个方法,就不会进入if语句中,而直接返回instance了
    }

    private SingletonLazy(){

    }
}
public class ThreadDemon29 {
    public static void main(String[] args) {
        SingletonLazy s1 = SingletonLazy.getInstance();
        SingletonLazy s2 = SingletonLazy.getInstance();
        System.out.println(s1 == s2);
    }

}

3:代码分析

很明显的不同是,初始,instance初始化为null,并没有实例化对象,只有第一次调用了getInstance方法后才会实例化SingletonLazy

4:“懒汉”的优点

例如现在有一个非常大的文件(1个G)

“饿汉”模式会直接把一个G全部加载到内存中,在进行展示。

“懒汉”模式会先加载100kb的数据到内存中,随着用户的翻页,在逐步的加载到内存当中

5:比较“饿汉”和“懒汉”

(1)创建时机上“饿汉”更早

(2)“饿汉”创建实例依赖于程序驱动,“懒汉”创建实例依赖于调用方法

四:“饿汉”模式线程安全问题

对于饿汉模式,无论有多少个线程在调用getInstance 方法,都会返回instance,对于return这一条代码来说,只有一个“读操作”,线程是非常安全的

五:“懒汉”模式的线程安全问题

1:重复创建实例

①看下面这个例子,我们拆分if里面的代码,会发现实例被new了两次,这就不是单例模式了,就有bug了

②有人说:不就是多new了个对象嘛,问题不大~~~。但是对于单例模式来说,一个对象可能要管理10GB的内存,或者更大,多new一个对象,就多翻了一倍的内存啊!!

2:解决问题

想办法给if条件和创建实例进行打包——用关键字synchronized。

(1)解决思路的核心本质:

是让,if语句打包成一个整体,成为“原子性操作”

(2)缺点:

①效率降低:

以下加锁的思路有一个致命缺点,我们保证多线程下我们创建出来的实例就只有一个是通过“加锁”和“解锁”来解决的,但是这个过程很耗费资源

②不可预期:

一旦线程发生了阻塞,那么什么时候解除阻塞我们是不可控制和预期的

3:优化效率问题

(1)问题解释

(2)解决方法

六:指令重排序问题

前引:

指令重排序是编译器优化的一种方式,调整原有代码的执行顺序,保证逻辑不变的前提下,提高程序的效率

1:代码拆分三指令

上面这段代码可以分为三个指令来执行 

①申请一段内存空间                    (买房子)

②在这段内存上,调用构造方法,初始化实例       (给房子装修)

③把内存地址赋值给instance                             (拿到房子的钥匙)

package thread;

/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: Hua YY
 * Date: 2024-09-24
 * Time: 13:52
 */
class SingletonLazy{
    private static Object locker1 = new Object();
    private static SingletonLazy instance = null;
    public static SingletonLazy getInstance(){
        if (instance == null){
            synchronized(locker1){
                if (instance == null){   //如果instance为null,则进入if创建实例,创建实例的时机是第一次调用getInstance这个方法
                    instance = new SingletonLazy();
                }
            }
        }

        return instance;//第二次第三次调用getInstance()这个方法,就不会进入if语句中,而直接返回instance了
    }

    private SingletonLazy(){

    }
}
public class ThreadDemon29 {
    public static void main(String[] args) {
        SingletonLazy s1 = SingletonLazy.getInstance();
        SingletonLazy s2 = SingletonLazy.getInstance();
        System.out.println(s1 == s2);
    }

}

2:分析

我们延用上面的代码

上述三个指令的执行顺序可以为①②③,也可以为①③②

问题出现在如果执行顺序为①③②上

3:代码解释

线程t1经过代码优化后,先执行③赋值地址,此时instance只是接收了一个地址,它所指向的是一个尚未被初始化的对象,这个对象啥也不是,可以理解成“全为0”

恰巧此时线程t2插入,进入if条件判断,判断instance != null,线程t2以为实例已经创建完毕,直接return instance,假设t2紧接着调用instance里面的方法或者属性,就会出现大问题。

4:解锁思路

加入关键字volatile

(1)复习volatile两个功能

①保证内存可见性(即每次访问变量都要重新读取内存,而不会优化到寄存器或者缓存当中)

②禁止指令重排序

此时针对变量的读写操作,就不会出现重排序的问题了

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Enhancing Trust in LLMs: Algorithms for Comparing and Interpreting LLMs

文章目录 题目摘要引言透明度的必要性对信任的追求困惑度测量自然语言处理(NLP)评估指标零投学习绩效少量学习性能迁移学习对抗测试公平和偏见稳健性评估LLMMaps基准测试和排行榜分层分析布鲁姆分类法的可视化幻觉评分知识分层策略利用机器学习模型进行层级生成注意力可视化LLM…

css五种定位总结

在 CSS 中,定位(Positioning)主要有五种模式,每种模式的行为和特点不同,以下是 static、relative、absolute、fixed 和 sticky 五种定位方式的对比总结: 1. static(默认定位) 特性…

阿里云函数计算 x NVIDIA 加速企业 AI 应用落地

作者:付宇轩 前言 阿里云函数计算(Function Compute, FC)是一种无服务器(Serverless)计算服务,允许用户在无需管理底层基础设施的情况下,直接运行代码。与传统的计算架构相比,函数…

【2023工业3D异常检测文献】PointCore: 基于局部-全局特征的高效无监督点云异常检测器

PointCore: Efficient Unsupervised Point Cloud Anomaly Detector Using Local-Global Features 1、Background 当前的点云异常检测器可以分为两类: (1)基于重建的方法,通过自动编码器重建输入点云数据,并通过比较原…

07-阿里云镜像仓库

07-阿里云镜像仓库 注册阿里云 先注册一个阿里云账号:https://www.aliyun.com/ 进入容器镜像服务控制台 工作台》容器》容器服务》容器镜像服务 实例列表》个人实例 仓库管理》镜像仓库》命名空间》创建命名空间 仓库管理》镜像仓库》镜像仓库》创建镜像仓库 使…

【AI】深度学习的数学--核心公式

1 梯度下降 f ( x Δ x , y Δ y ) ≃ f ( x , y ) ∂ f ( x , y ) ∂ x Δ x ∂ f ( x , y ) ∂ y Δ y f(x\Delta x,y\Delta y) \simeq f(x,y)\frac{\partial f(x,y)}{\partial x}\Delta x\frac{\partial f(x,y)}{\partial y}\Delta y f(xΔx,yΔy)≃f(x,y)∂x∂f(x,y)​…

MySQL 性能剖析全攻略

在使用 MySQL 数据库的过程中,性能问题往往是让开发者和管理员头疼的难题。为了有效地解决这些问题,我们需要对 MySQL 进行性能剖析。那么,如何在 MySQL 中进行性能剖析呢?本文将为你详细介绍。 一、为什么要进行性能剖析&#x…

基于安卓开发大型体育场管理系统的设计与实现(源码+定制+讲解)

博主介绍: ✌我是阿龙,一名专注于Java技术领域的程序员,全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师,我在计算机毕业设计开发方面积累了丰富的经验。同时,我也是掘金、华为云、阿里云、InfoQ等平台…

《开题报告》基于SpringBoot框架的高校专业实习管理系统开题报告的设计与实现源码++学习文档+答辩讲解视频

开题报告 研究背景 在当今高等教育日益普及与深化的背景下,高校专业实习作为学生将理论知识转化为实践能力、提前适应社会工作环境的重要环节,其重要性不言而喻。然而,传统的高校专业实习管理模式往往存在信息不对称、流程繁琐、效率低下、…

SSM+Vue共享单车管理系统

目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 spring-mybatis.xml3.5 spring-mvc.xml3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍:CSDN认证博客专家,CSDN平台Java领域优质创作…

代码随想录_刷题记录_第四次

二叉树 — 理论基础 种类: 满二叉树(所有层的节点都是满的,k:深度 节点数量:2^k - 1)完全二叉树(除了最后一层,其余层全满,并且最后一层从左到右连续)二叉搜…

信道衰落的公式

对于天线: 对于天线的面积计算: 天线的接收功率密度: 天线的接收功率: 移动无线信道(I) (xidian.edu.cn)https://web.xidian.edu.cn/zma/files/20150710_153736.pdf 更加常用的考虑了额外的信道衰落pathlo…

2024 maya的散布工具sppaint3d使用指南

目前工具其实可以分为三个版本 1 最老的原版 时间还是2011年的,只支持python2版的maya 2 作者python3更新版 后来作者看maya直到2022上还是没有类似好用方便的工具,于是更新到了2022版本 这个是原作者更新的2022版本,改成了python3&#…

敏感字段加密 - 华为OD统一考试(E卷)

2024华为OD机试(E卷+D卷+C卷)最新题库【超值优惠】Java/Python/C++合集 题目描述 【敏感字段加密】给定一个由多个命令字组成的命令字符串: 1、字符串长度小于等于127字节,只包含大小写字母,数字,下划线和偶数个双引号; 2、命令字之间以一个或多个下划线 进行分割; 3、可…

Study-Oracle-10-ORALCE19C-RAC集群搭建(一)

一、硬件信息及配套软件 1、硬件设置 RAC集群虚拟机:CPU:2C、内存:10G、操作系统:50G Openfile数据存储:200G (10G*2) 2、网络设置 主机名公有地址私有地址VIP共享存储(SAN)rac1192.168.49.13110.10.10.20192.168.49.141192.168.49.130rac2192.168.49.13210.10.10.3…

单体到微服务架构服务演化过程

架构服务化 聊聊从单体到微服务架构服务演化过程 单体分层架构 在 Web 应用程序发展的早期,大部分工程是将所有的服务端功能模块打包到单个巨石型(Monolith)应用中,譬如很多企业的 Java 应用程序打包为 war 包,最终会形…

JSP(Java Server Pages)基础使用二

简单练习在jsp页面上输出出乘法口诀表 既然大家都是来看这种代码的人了&#xff0c;那么这种输出乘法口诀表的这种简单算法肯定是难不住大家了&#xff0c;所以这次主要是来说jsp的使用格式问题。 <%--Created by IntelliJ IDEA.User: ***Date: 2024/7/18Time: 11:26To ch…

线性表二——栈stack

第一题 #include<bits/stdc.h> using namespace std; stack<char> s; int n; string ced;//如何匹配 出现的右括号转换成同类型的左括号&#xff0c;方便我们直接和栈顶元素 char cheak(char c){if(c)) return (;if(c]) return [;if(c}) return {;return \0;/…

css边框修饰

一、设置线条样式 通过 border-style 属性设置&#xff0c;可选择的一些属性如下&#xff1a; dotted&#xff1a;点线 dashed&#xff1a;虚线 solid&#xff1a;实线 double&#xff1a;双实线 效果如下&#xff1a; 二、设置边框线宽度 ① 通过 border-width 整体设置…

【深度学习】深度卷积神经网络(AlexNet)

在 LeNet 提出后&#xff0c;卷积神经网络在计算机视觉和机器学习领域中很有名气&#xff0c;但并未起到主导作用。 这是因为 LeNet 在更大、更真实的数据集上训练的性能和可行性还有待研究。 事实上&#xff0c;在 20 世纪 90 年代到 2012 年之间的大部分时间里&#xff0c;…