单例模式的相关知识

news2025/1/24 5:24:02

饿汉模式

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.coloradmin.cn/o/929733.html

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

相关文章

Sentinel dashboard无法查询到应用的限流配置问题以及解决

一。问题引入 使用sentinle-dashboard控制台 项目整体升级后,发现控制台上无法看到流控规则了 之前的问题是无法注册上来 现在是注册上来了。结果看不到流控规则配置了。 关于注册不上来的问题,可以看另一篇文章 https://blog.csdn.net/a15835774652/…

Mybatis与Spring集成配置

目录 具体操作 1.1.添加依赖 1.2创建spring的配置文件 1.3. 注解式开发 Aop整合pagehelper插件 1. 创建一个AOP切面 2. Around("execution(* *..*xxx.*xxx(..))") 表达式解析 前言: 上篇我们讲解了关于Mybatis的分页,今天我们讲Mybatis与…

如何向BertModel增加字符

这里写自定义目录标题 看起来add_special_tokens和add_tokens加入的新token都不会被切分。

paddleclas ImportError: cannot import name ‘Identity‘ from ‘paddle.nn‘

使用paddlepaddle的 paddleclas 官方demos时 ,报错如图 ImportError: cannot import name ‘Identity’ from ‘paddle.nn’ 解决方案很简单: 找到调用 Identity 的位置: 注释掉就解决啦 !!! 搞定!!!…

4.14 tcp_tw_reuse 为什么默认是关闭的?

开启 tcp_tw_reuse 参数可以快速复用处于 TIME_WAIT 状态的 TCP 连接时,相当于缩短了 TIME_WAIT 状态的持续时间。 tcp_tw_reuse 是什么? TIME_WAIT 状态的持续时间是 60 秒,这意味着这 60 秒内,客户端一直会占用着这个端口。端…

使用训练工具

HuggingFace上提供了很多已经训练好的模型库,如果想针对特定数据集优化,那么就需要二次训练模型,并且HuggingFace也提供了训练工具。 一.准备数据集 1.加载编码工具 加载hfl/rbt3编码工具如下所示: def load_encode():# 1.加载编…

用C/C++修改I2C默认的SDA和SCL针脚

首先要说明一点:Pico 有两个 I2C,也就是两套 SDA 和 SCL。这点你可以在针脚图中名字看出,比如下图的 Pin 4 和 Pin 5是 I2C1 的,而默认的 Pin 6 和 Pin 7 是 I2C0 的。 默认情况下是只开启了第一个 I2C,也就是只有 I2C…

数据库——缓存数据

文章目录 缓存数据的处理流程是怎样的?为什么要用 Redis/为什么要用缓存? 缓存数据的处理流程是怎样的? 简单来说就是: 如果用户请求的数据在缓存中就直接返回。缓存中不存在的话就看数据库中是否存在。数据库中存在的话就更新缓存中的数据。…

基于云原生网关的流量防护实践

作者:涂鸦 背景 在分布式系统架构中,每个请求都会经过很多层处理,比如从入口网关再到 Web Server 再到服务之间的调用,再到服务访问缓存或 DB 等存储。在下图流量防护体系中,我们通常遵循流量漏斗原则进行流量防护。…

数字孪生赋能工业制造,为制造业带来新机遇与挑战

数字孪生技术是利用模拟仿真技术将实体对象数字化的技术。它基于虚拟现实、人工智能和云计算等技术,能够创建与真实物体相同的数字模型,并通过实时监测和分析手段,为制造企业提供关于该物体的全面数据,从而优化产品开发和生产过程…

《Dive into Deep Learning》

《Dive into Deep Learning》:https://d2l.ai/ Interactive deep learning book with code, math, and discussionsImplemented with PyTorch, NumPy/MXNet, JAX, and TensorFlowAdopted at 500 universities from 70 countries 《动手学深度学习》中文版&#xff1…

dji uav建图导航系列()ROS中创建dji_sdk节点包(一)项目结构

文章目录 1、整体项目结构1.1、 目录launch1.2、文件CMakeLists.txt1.3、文件package.xml1.4、目录include1.4、目录srv在ROS框架下创建一个无人机的节点dji_sdk,实现必需的订阅(控制指令)、发布(无人机里程计)、服务(无人机起飞降落、控制权得很)功能,就能实现一个类似…

C#-集合小例子

目录 背景: 过程: 1.添加1-100数: 2.求和: 3.平均值: 4.代码:​ 总结: 背景: 往集合里面添加100个数,首先得有ArrayList导入命名空间,这个例子分为3步,1.添加1-100个数2.进行1-100之间的总和3.求总和的平均值&…

03.sqlite3学习——数据类型

目录 sqlite3学习——数据类型 SQL语句的功能 SQL语法 SQL命令 SQL数据类型 数字类型 整型 浮点型 定点型decimal 浮点型 VS decimal 日期类型 字符串类型 CHAR和VARCHAR BLOB和TEXT SQLite 数据类型 SQLite 存储类 SQLite 亲和类型(Affinity)及类型名称 Boo…

【微服务】04-Polly实现失败重试和限流熔断

文章目录 1. Polly实现失败重试1.1 Polly组件包1.2 Polly的能力1.3 Polly使用步骤1.4 适合失败重试的场景1.5 最佳实践 2.Polly实现熔断限流避免雪崩效应2.1 策略类型2.2 组合策略 1. Polly实现失败重试 1.1 Polly组件包 PollyPolly.Extensions.HttpMicrosoft.Extensions.Htt…

MaBatis中的分页插件以及特殊字符处理

目录 一、PageHelper介绍 二、PageHelper使用 1. 导入pom依赖 2. Mybatis.cfg.xml 配置拦截器 配置sql映射文件 测试代码 特殊字符处理 2. 使用CDATA 区段 一、PageHelper介绍 PageHelper 是 Mybatis 的一个插件,这里就不扯了,就是为了更加便捷的进…

记录一次“top负1”比赛经历

获奖啦! 比赛题目:中文语义病句识别与纠正挑战赛 比赛链接:https://challenge.xfyun.cn/topic/info?typeidentification-and-correction&optionphb“请介绍你们团队” “各位评委老师,我是来自WOT团队的选手AMBT&#xff0…

Python|爬虫和测试|selenium框架的安装和初步使用(一)

前言: Python作为一门胶水语言来说,可以说是十分的优秀,什么事情都可以干,并且在某些领域还能干的非常不错,尤其是在爬虫和测试领域,该语言可以说是没有对手。 这么说的原因是因为如果你要使用爬虫爬取某…

4.网络设计与redis、memcached、nginx组件(二)

系列文章目录 第四章 网络设计与redis、memcached、nginx组件(一) 第五章 网络设计与redis、memcached、nginx组件(二) 文章目录 系列文章目录[TOC](文章目录) 前言一、reactor模型?二、Reactor 开发1.建立连接 三、典型reactor 模型单reactor 模型典型 readisradi…

C++避坑——most vexing parse问题

1."坑"的问题是什么&#xff1f; 先看一段代码&#xff1a; class Functor { public:void operator()(){std::cout << "我是线程的初始函数" << std::endl;} };int main() {std::thread t(Functor());// 强制高速编译器这是一个构造函数!t.j…