多线程环境下的单例模式

news2024/12/24 10:24:32

✨✨hello,愿意点进来的小伙伴们,你们好呐!
🐻🐻系列专栏:【JavaEE初阶】
🐲🐲本篇内容:基于多线程的单例模式
🐯🐯作者简介:一名现大二的三非编程小白,日复一日,仍需努力。

单例模式是设计模式中很常见的一种,属于大佬们为了像我这种小菜鸟能够将代码写得水平好像还可以 , 针对一些经典的场景 , 发明出来的一种"棋谱",给出的一些典型的解决方案中的一种

单例模式分为 饿汉式 和 懒汉式 两种
在某些场景下,就需要对一个类只能创建一个实例,那么像这种需求,单例模式就应运而生了,使用了单例模式后,想创建多个实例都要费尽心思也没办法创建,

饿汉式 :

public class Singleton1 {
    private static Singleton1 instance = new Singleton1();

    public static Singleton1 getInstance() {
        return instance;
    }
    
    private Singleton1() {}
}

🎠🎠🎠

上述代码就是单例模式中的饿汉模式:

  1. 在类中先将类的实例对象创建出来,且赋给了静态的属性,让这个属性在类加载的时候就创建存在了,且这个类只有一份类属性,所以保证了这个实例对象只有一份.
  2. 如果想要获取到该对象,就使用get方法,该方法直接返回以及创建好的静态对象.统一提供Singleton1.getInstance() 来获取该实例对象.
  3. 为了避免 Singleton1 类不小心被复制出多份,把构造方法给私有化,所以在类的外部就无法创建该类的对象了

🎈🎈🎈
饿汉模式使用start保证了这个实例的唯一,然后再类加载的时候就创建该类的实例对象,将构造方法私有化,保证在类的外部无法new这个对象;
该模式被称为饿汉式的原因可以理解为:该模式在类加载的时候就创建了对象,显得很急切,很饿的感觉,所以将其称为饿汉模式.

懒汉式 :

public class Singleton2 {
    private static Singleton2 instance = null;

    public static Singleton2 getInstance() {
        if(instance == null) {
            instance = new Singleton2();
        }
        return instance;
    }

    private Singleton2() {}
}

🎡🎡🎡

上述代码就是单例模式中的懒汉式:

  1. 懒汉式对比饿汉式的区别就是创建对象的时机,饿汉式创建对象是在加载类的时候创建的,而懒汉式在类加载的时候是定义了类的属性,然后这个属性指向null;
  2. 然后在get方法中,判断是否这个属性为null,如果是第一次要使用该类的实例,对象为null就创建对象,不为null就直接返回之前创建好的类实例属性.
  3. 构造方法也是私有化处理

✨✨✨
懒汉式的创建对象是在要用再来创建,这样子的模式似乎有点懒,但是对于程序来说,懒是褒义词,懒说明资源不会浪费很严重,这样子的创建模式可以避免在用不上对象实例的时候,并不会创建对象实例,避免了资源的浪费

多线程场景下的单例模式 :

上面讲的两个单例模式是否能经得起多线程的敲打呢?在日常环境下,大多数都是多线程场景;

🎃🎃🎃
先来说说饿汉式吧;
在这里插入图片描述

在饿汉模式下,类的实例对象是在类加载阶段就创建的且Java中规定只能创建一个类属性,然后饿汉式代码中就只是对该类属性进行从主内存中读取到线程的本地内存中,然后线程访问到进行返回类实例对象.
这种情况下操作是原子操作,无法继续分解了,然后该实例对象是一开始就存在的,所以也并不存在内存可见性的情况;
所以在饿汉式中,代码是线程安全的

🎄🎄🎄
接下来来看看懒汉式:

public class Singleton2 {
    private static Singleton2 instance = null;

    public static Singleton2 getInstance() {
        if(instance == null) {
            instance = new Singleton2();
        }
        return instance;
    }

    private Singleton2() {}
}

在懒汉式的代码中,最开始类加载阶段创建的是一个类属性,然后这个属性指向是null,然后在get方法中,该类是先将类属性从主内存中读取(load),然后再进行判断(cmp),然后依照判断后的操作进行new <<先申请一个内存可见,调用构造方法将这个内存初始化为一个合理的对象,把内存地址空间赋给类属性的引用 >>,然后写入内存(save),然后返回实例对象;或者直接返回
上面的操作虽然每一步都是原子操作,但是其实都是非原子操作分解出来的,所以懒汉式其实是会引起线程安全问题的.

下面我们来看看怎么避免懒汉式中的线程安全问题>>

解决懒汉式的线程安全问题:

public class Singleton2 {
    private static Singleton2 instance = null;

    public static Singleton2 getInstance() {
        if(instance == null) {
            instance = new Singleton2();
        }
        return instance;
    }

    private Singleton2() {}
}

这个是单线程下的懒汉式代码,我们来分析一下会有什么原因引发的线程安全

在这里插入图片描述

因为操作的非原子性,所以CPU的调度会将其分为原子操作去执行,最后会导致我们不想要的执行结果,最后导致了线程安全问题.
所以下面,我们先来解决原子操作的问题;

public class Singleton2 {
    private static Singleton2 instance = null;

    public static Singleton2 getInstance() {
        synchronized (Singleton2.class) {
            if(instance == null) {
                instance = new Singleton2();
            }
        }
        return instance;
    }
    private Singleton2() {}
}

加锁后会将该操作视为原子操作:

在这里插入图片描述

但是这样子的加锁方式其实会导致资源的浪费,因为这样子的加锁会导致每一次调用get方法的时候都会加锁,然后释放锁,其实是没有必要的,加锁操作在第一次要创建该对象实例的时候加锁就行了.

public class Singleton2 {
    private static Singleton2 instance = null;

    public static Singleton2 getInstance() {
       if(instance == null) {
           synchronized (Singleton2.class) {
               if(instance == null) {
                   instance = new Singleton2();
               }
           }
       }
        return instance;
    }
    private Singleton2() {}
}

优化到了什么的代码,上面的代码中get方法里,第一个if判断是为了判断是否是第一次创建对象,需不需要加锁,第二个if判断是判断需不需要创建对象;
这样子的两个判断可以有效地避免对加锁操作的频繁使用,只对第一次创建对象加锁即可.

但是这样子的代码其实也是有问题的 ! 内存可见性问题 ! ! ! 指令重排序问题 ! ! !
要是很多线程都去进行 getinstance() ,这个时候,有可能会有JVM优化的风险,最后可能导致只有第一次读取才是真正的内存的值,后面的读取都是读线程本地内存的值.导致内存可见性问题

也有可能因为编译器的优化指令重排序,导致了new这个执行步骤的优化,最后也会产生线程安全问题.

所以我们在属性中加了 volatile 来避免出现内存可见性和指令重排序的两个问题;

public class Singleton2 {
    private volatile static Singleton2 instance = null;

    public static Singleton2 getInstance() {
       if(instance == null) {
           synchronized (Singleton2.class) {
               if(instance == null) {
                   instance = new Singleton2();
               }
           }
       }
        return instance;
    }
    private Singleton2() {}
}

🎍🎍🎍这个时候我们的代码才优化完成,适用于多线程环境下的懒汉式的单例模式

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux虚拟化网络之路由配置

一、Linux路由配置 如果要在不同网段直接通讯&#xff0c;需要添加路由&#xff0c;Linux添加路由命令如下&#xff1a; route [add|del] [-net|-host] target [netmask Nm] [gw Gw] [[dev] If] add : 添加一条路由规则&#xff1b;del : 删除一条路由规则&#xff1b;-net …

Win11如何开启移动中心页面的操作方法教学

Win11如何开启移动中心页面的操作方法教学分享。有用户不知道怎么去打开移动中心&#xff0c;开启这个页面我们可以去进行屏幕亮度调整、声音调整、笔记本电池状态、外接显示器/投影仪、以及幻灯片显示模式等功能集中到一个面板上进行管理设置。如何开启这个页面&#xff0c;来…

【教程】超详细通过Shizuku转生支付宝集成XQ_Crystal来自动收能量

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.blog.csdn.net] 通过Shizuku是比应用转生更好更稳定的方法&#xff01; 可以先看这篇&#xff1a;免Root使用Xposed插件并开启蚂蚁森林自动偷能量,比应用转生好 还不会的&#xff0c;继续往下。看完还不会&#xff…

手机银行APP评测系列:天津银行持续优化手机银行用户体验,但仍需加强细节提升

易观分析&#xff1a;作为银行金融服务线上场景渗透的有效抓手&#xff0c;当前手机银行APP已经成为其触达用户的重要渠道。随着银行发力场景服务平台成为发展趋势&#xff0c;5G技术问世对金融服务场景端提出新要求&#xff0c;用户体验反馈成为银行线上场景化运营的重要一环。…

JavaScript—分支结构和循环结构整理

目录 一、流程控制 二、分支结构 1. if语句 2. if…else语句 3. if…else if语句 4. switch语句 5. 条件表达式构成的选择结构 三、循环结构 1.while循环 2. do-while循环 3. for循环 3.1 for循环转换为while循环 3.2 断点调试 4. 循环嵌套 JavaScript 是一种解释…

微服务框架 SpringCloud微服务架构 16 SpringAMQP 16.6 FanoutExchange

微服务框架 【SpringCloudRabbitMQDockerRedis搜索分布式&#xff0c;系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 SpringCloud微服务架构 文章目录微服务框架SpringCloud微服务架构16 SpringAMQP16.6 FanoutExchange16.6.1 发布订阅 - Fanout Exchange16.6…

火山引擎DataTester:一个爆款游戏产品,是如何用A/B测试打磨出来的?

更多技术交流、求职机会&#xff0c;欢迎关注字节跳动数据平台微信公众号&#xff0c;回复【1】进入官方交流群 随着国内游戏用户数量趋于饱和&#xff0c;中国游戏产业也从高速成长期逐渐转型&#xff0c;市场成熟度提升&#xff0c;竞争趋于精细化。 随着游戏出海以及私域流…

Java开发学习(二十六)----SpringMVC返回响应结果

SpringMVC接收到请求和数据后&#xff0c;进行了一些处理&#xff0c;当然这个处理可以是转发给Service&#xff0c;Service层再调用Dao层完成的&#xff0c;不管怎样&#xff0c;处理完以后&#xff0c;都需要将结果告知给用户。 比如:根据用户ID查询用户信息、查询用户列表、…

2022世界杯La‘eeb肖像,python海龟实现啦

努力是为了不平庸~ 学习的最大理由是想摆脱平庸&#xff0c;早一天就多一份人生的精彩&#xff1b;迟一天就多一天平庸的困扰。 目录 一、Laeeb的身世 二、开始作画 1、准备阶段 2、常用指令 3、开始做画 一、Laeeb的身世 2022年卡塔尔世界杯&#xff08;英语&#xff1a…

华为机试 - 高效的任务规划

目录 题目描述 输入描述 输出描述 用例 题目解析 算法源码 题目描述 你有 n 台机器&#xff0c;编号为 1~n&#xff0c;每台都需要完成一项工作&#xff0c;机器经过配置后都能完成独立完成一项工作。假设第 i 台机器你需要花 B 分钟进行设置&#xff0c;然后开始运行&a…

[附源码]计算机毕业设计室内设计类网站Springboot程序

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

Azure考试认证经验

目录 一、概述 1、考试报名 2、考试经验 3、Azure居家考试 4、Azure证书 一、概述 人不管做什么事情&#xff0c;决心很重要。有了目标就要坚定不移的去执行&#xff0c;这是最基本的。如果没有一个正确的心态来学习&#xff0c;建议你放弃。 1、考试报名 Azure考试报名…

2022 年度优秀开源技术

2022 年&#xff0c;我们综合了 OSCHINA 平台上各大认证官方技术团队、开源社区帐号年度发表的内容频率及质量、开展各种活动运营积极性等多方面的表现 具体名单如下&#xff08;按首字母顺序排名&#xff0c;不分先后&#xff09;&#xff1a; Alluxio 官方 …

osgEarth示例分析——osgearth_cluster

前言 osgearth_cluster示例&#xff0c;展示了绘制很多模型时&#xff0c;如何合适设置模型遮挡的混乱情况。 当模型过多时&#xff0c;可以创建 osgEarth::Util::ClusterNode 节点对象&#xff0c;然后创建 osg::NodeList&#xff0c;将需要绘制的节点 node 们&#xff0c;都…

[附源码]计算机毕业设计JAVA影院售票系统

[附源码]计算机毕业设计JAVA影院售票系统 项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis M…

Mockito verify Junit5集成 Mockito

Mockito 集成 Junit5 在学习Mockito 如何集成 Junit5 之前&#xff0c;先来学习下 Mockito 基础的verify功能。 Maven依赖 本篇博客代码的Maven依赖如下&#xff0c;源码地址 <dependencies><dependency><groupId>org.springframework</groupId>&l…

分享107个小清新ppt模板,总有一款适合您

PPT下载链接&#xff1a;https://pan.baidu.com/s/1WqaR_29avEgq46iTSLKfmw?pwd5r81 提取码&#xff1a;5r81 源码下载链接&#xff1a;ppt.rar - 蓝奏云 采集的参数 page_count 1 # 每个栏目开始业务content"text/html; charsetgb2312"base_url "https:…

大一学生《Web编程基础》期末网页制作 基于HTML+CSS+JavaScript响应式个人主页相册介绍模板

&#x1f389;精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; ✍️ 作者简介: 一个热爱把逻辑思维转变为代码的技术博主 &#x1f482; 作者主页: 【主页——&#x1f680;获取更多优质源码】 &#x1f393; web前端期末大作业…

UNIX环境高级编程_文件IO_文件描述表

这篇文件记录文件描述表。 2 文件描述符表 2.1 什么是文件描述符表 当运行一个程序时&#xff0c;内核会创建进程表task_struct。当open 打开文件后&#xff0c;会在进程表中创建相应的结构体来记录打开的文件&#xff0c;这个结构体就是文件描述符表。 2.2 task_struct 与…

Apple M1 开启HiDPI的新方法,无需虚拟屏,无需SwitchResX

之前折腾2K屏开HiDPI时记录过三种方法&#xff1a; 2020年03月&#xff1a;SwitchResX 开启HiDPI时显示Not installed的解决办法 2020年11月&#xff1a;升级macOS Big Sur 后 HIDPI失效的解决办法 2021年12月&#xff1a; Apple M1 开启HiDPI的新方法&#xff0c;无需关闭SIP&…