单例模式的相关知识

饿汉模式

package Thread;
class Singleton{private static Singleton instance = new Singleton();public static Singleton getInstance(){return instance;}private Singleton(){}
}public class demo1 {public static void main(String[] args) {Singleton S1 =Singleton.getInstance();Singleton S2 =Singleton.getInstance();System.out.println(S1==S2);
//        Singleton S3 =new Singleton();}
}

在这里插入图片描述

所谓的单例模式,就是指一个类的对象只能创建一个,上述代码中,Singleton这个类中,有一个static修饰的对象,这个对象就是一个类对象,具有类属性,由于类对象只有一个,也就是说,每个类的类对象是单例的,所以,类对象 的属性(static),也就是单例的了,这个对象创建的时间是在Singleton第一次被JVM使用的时候.
有了这个对象后,不能允许在类外面创建对象,因此,我们就需要将类的构造方法改为private修饰,这样,只要在类外面尝试用new来创建对象,而不是调用get方法,系统就会报错.
在这里插入图片描述
在这里插入图片描述
但是,这样并不保证就一定不可以在类外面创建多个不同的对象,"反射"这个操作就可以做到,但是做这个操作需要谨慎,滥用反射,会带来极大的风险,因此,我们并不推荐使用这个方法.

懒汉模式

package Thread;
class Singletonlazy{private static Singletonlazy instance = null;public static Singletonlazy getInstance(){if(instance==null){instance = new Singletonlazy();}return instance;}private Singletonlazy(){}
}
public class demo2 {public static void main(String[] args) {Singletonlazy S1 = Singletonlazy.getInstance();Singletonlazy S2 = Singletonlazy.getInstance();System.out.println(S1==S2);}
}

在这里插入图片描述
懒汉模式与饿汉模式最主要的区别就是,类对象创建的时间不同,懒汉模式并不是Singletonlazy类在第一次被加载的时候创建的,而是在get方法第一次被调用的时候才会创建

线程安全问题

饿汉模式的get方法
在这里插入图片描述
懒汉模式的get方法
在这里插入图片描述
从上述代码中可以看出,饿汉模式下的get方法只涉及到读取操作,并没有进行对变量的修改,而懒汉模式下的get方法,涉及对对象引用的修改,而且是存在if的判断条件,说明该修改操作不是一个原子级别的,综上所述,懒汉模式存在严重的线程安全问题,会导致出现两个线程都new出一个新的对象的情况,虽然系统最后会对后面new出来的对象进行清除,但是,创建两个对象所花费的资源和时间确是巨大的损失,所以,为了避免这样的损失,我们需要对该方法进行加锁

加锁

为了解决懒汉模式下的线程安全问题,我们尝试对get方法的修改操作进行加锁,注意,这里进行加锁的范围需要包含if的条件判断和修改的具体操作,需要将这两个步骤所在一起,才能解决线程安全问题
但是我们也需要注意,加锁也不能滥用,因为加锁也是很消耗资源和时间,能不加锁的情况就不要加锁,否则,会对代码的运行效率造成极大的影响
就比如这里我们锁需要加锁的代码,只有第一次调用的时候,才会对对象的引用进行修改,后续再次调用该方法的时候,也只是如饿汉模式下的方法一样,只是进行读操作,所以,我们要求,在加锁的时候,我们也要做到,只有在第一次具有线程安全问题的情况下,我们才进行加锁,其余情况下,就避免加锁带来 的额外的负担,那我们需要怎么做呢
再加一个if判断
在这里插入图片描述
先判断是否需要加锁,再判断是否真的需要加锁
第一个if是为了判断是否需要加锁,第二个if是判断是否需要创建新的对象
因为这里是多线程,在加锁的时候,可能涉及到阻塞的情况,阻塞时间的长短我们也不知道,因此,可能存在由于两个if之间的时间相隔过久,在这段时间里,可能会有其他线程将对象给修改掉的情况
但是这里的第二个线程一定能读取到第一个线程修改的值吗,这里就涉及到之前章节讲到的内存可见性

内存可见性

为了避免编译器优化导致的系统bug,我们需要给instance这个变量加上一个volatile,来确保代码的准确性
在这里插入图片描述

指令重排序

这里除了可能会有内存可见性的问题之外,也可能会有指令重排序导致的问题出现,什么是指令重排序呢,举个例子
假如我们完成一个工作的步骤是123,经过编译器优化之后,步骤就变为了132,甚至321.这样的优化的好处和内存可见性的优化一样,都可以大幅度提高程序运行的效率,但是这如果只是在单线程的情况下还好,但是在多线程的情况下了,就有很大可能会引发线程安全问题,再举个例子
我们想要new一个instance对象出来,需要分为三个步骤
1.创建出内存空间,得到内存地址
2.再空间上调用构造方法,对对象进行初始化
3.把内存地址幅值给Instance引用
如果是单线程的情况下,23可以互调,不会有太大的影响
但是如果发生在多线程的情况下,此时还没来得及给instance初始化,就调度给其他线程了,第二个线程在执行的时候,判定instance!=null,于是就把instance给返回出去了,并且后续可能还会涉及掉用instance中的方法,而instance 现在只是一具空壳,并没有被初始化,这就可能会造成严重的问题
那么我们怎么解决这个问题呢,这里我们也可以利用volatile来解决问题.
给instance加上volatile之后,就不会产生指令重排序了

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为数通方向HCIP-DataCom H12-821题库(单选题:61-80)

第61题 关于 BGP 的Keepalive报文消息的描述,错误的是 A、Keepalive周期性的在两个BGP邻居之间发送 B、Keepalive报文主要用于对等路由器间的运行状态和链路的可用性确认 C、Keepalive 报文只包含一个BGP数据报头 D、缺省情况下,Keepalive 的时间间隔是180s 答案&#xff…

csdn冷知识:如何在csdn里输入公式或矩阵

目录 1 输入公式 2 输入矩阵 3 如何输入复杂公式 4 如何修改,已经生成的公式 1 输入公式 进入编辑模式点击右边的菜单:公式然后进入公式编辑器,选择右边的 ... 可以选择大括号等,右边还有矩阵符号选择后你需要创建几行几列的…

《JVM修仙之路》初入JVM世界

《JVM修仙之路》初入JVM世界 博主目前正在学习JVM的相关知识,想以一种不同的方式记录下,娱乐一下 清晨,你睁开双眼,看到刺眼的阳光,你第一反应就是完了完了,又要迟到了。刚准备起床穿衣的你突然意识到不对&…

数字化、智能化的酒店固定资产管理系统

酒店固定资产管理系统是一种专门为酒店行业定制的管理软件,可以帮助酒店管理者全面、准确地管理固定资产。该系统具有以下实际功能和特点:  资产库存功能:通过扫描二维码或手动输入条形码,完成酒店固定资产的有效总结&#xff0…

嵌入式系统存储体系

一、存储系统概述 主要分为三种:高速缓存(cache)、主存和外存。 二、高速缓存Cache 高速缓冲存储器中存放的是当前使用得最多得程序代码和数据,即主存中部分内容的副本,其本身无自己的地址空间。在嵌入式系统中Cac…

刘强东的低价武器再上膛

电商的光辉正撒向中国的每一个角落。 河北南部的一个村子,74岁的陈奶奶站在大门口,正等待着小哥送货上门,孤身在家的她平时吃的油盐酱醋,喝的奶粉豆浆,都由女儿崔丽丽在网上买。由于京东是村子里唯二可以上门的快递&a…

在腾讯云服务器OpenCLoudOS系统中安装Jenkins(有图详解)

Jenkins介绍 Jenkins是一个开源软件项目,是基于java开发的一种持续集成工具,用于监控持续重复的工作,旨在提供一个开放易用的软件平台,使软件的持续集成变成可能。 将项目代码的svn地址配置在Jenkins,就可以直接在Je…

RISC-V 中国峰会 | OpenMPL引人注目,RISC-V Summit China 2023圆满落幕

RISC-V中国峰会圆满落幕 2023年8月25日,为期三天的RISC-V中国峰会(RISC-V Summit China 2023)圆满落幕。本届峰会以“RISC-V生态共建”为主题,结合当下全球新形势,把握全球新时机,呈现RISC-V全球新观点、新…

Oracle监听器启动出错:本地计算机上的OracleOraDb11g_home1TNSListener服务启动后又停止了解决方案

在启动oracle的服务OracleOraDb11g_home1TNSListener时,提示服务启动后又停止了。 解决方法: 修改oracle安装目录下的两个配置文件: 以上两个文件,对应的HOST的值,都改为127.0.0.1 然后再启动服务,启动成…

RISC-V公测平台发布 · 在SG2042上配置Jupiter+Octave科学计算环境

简介 JupyterHub是一个开源的共享计算平台,它为每个用户管理一个单独的 Jupyter 环境, 可以用于学生班级、企业数据科学小组或科学研究小组。它是一个多用户中心,可以生成、管理和代理多个单用户Jupyter笔记本服务器的实例。 GNU Octave是一…

C++中数组作为参数进行传递方法

文章目录 基础:数组作为函数形参示例:1、一维数组的传递(1)直接传递(2)指针传递(3)引用传递 2、二维数组的传递(1)直接传递(2)指针传递…

3d素材库素材资源平台大大节省老师备课时间

教育元宇宙相信大家有所耳闻,3D素材云库通过数字三维建模技术将现实中的物体、天气、灯光等1:1模拟还原到虚拟场景中,让人们在教育元宇宙平台中可视、可见、可感。 在元宇宙爆发的大背景下,3D互联网传播内容也将迎来一次全面升级&…