【单例模式】单例模式创建的几种方式

news2024/10/3 4:34:45

一、饿汉模式

饿汉模式是在类加载的时候就初始化了一份单例对象,所以他不存在线程安全问题。

优点:不存在线程安全问题,天然的线程安全

缺点:在类加载的时候就已经创建了对象,如果后续代码里没有使用到单例,就浪费了资源,通过反射还是可以去new的

public Class Single {
    private static Single single = new Single();

    private Single() {}

    public static Single getInstance() {
        return single;
    }
}

二、懒汉模式(处理线程安全)

懒汉模式顾名思义懒汉,没有提前去准备,而是用到的时候准备,比如洗碗,饿汉模式是吃完饭立马去洗,不管下次是点外卖还是家里吃(用到洗的碗)。懒汉模式就是,不要管下一顿从外卖还是家里做,啥时候用到碗啥时候洗。代码1如下

public class Single {
    private static Single single;
    
    private Single(){}
    
    public static Single getInstance(){
        // 如果是第一次获取就创建
        if (single == null){
            single = new Single();
        }
        return single;
    }
}

此时如果两个线程同时去调用了这个getInstance方法,两个线程线拿single去与null比较,当线程1比较之后if成立,此时线程2也比较成立都进入if,此时就new出来了两个对象,实例创建出来了多个。此时我们发现我们需要通过加锁来把这个比较与new操作打包成原子性的,代码2如下:

public class Single {
    private static Single single;

    private Single(){}

    public static Single getInstance(){
        // 加锁
        synchronized (Single.class){
            if (single == null){
                single = new Single();
            }
        }
        
        return single;
    }
}

此时我们就解决了上述两个线程同时进入if 里new出多个实例的问题,解决了实例还没有被创建时的线程安全问题,但是如果实例已经被创建好了之后,后续多个线程去调用获取实例时,依旧要去加锁,但是此时加锁的开销大但没有线程问题,所以此时我们需要在锁的外层再加一个判断,判断是第一次创建实例还是实例已经创建线程是直接调用。代码3如下:

public class Single {
    private static Single single;

    private Single(){}

    public static Single getInstance(){
        if (single == null){
            // 如果是第一次去创建实例,就加锁解决线程安全问题
            synchronized (Single.class){
                if (single == null){
                    single = new Single();
                }
            }
        }
            
        return single;
    }
}

上述代码解决了后续访问加无效锁开销大的问题,此时如果实例没有创建,有两个线程调用这个方法,线程1拿到了锁进入第二层if去new对象,new的本质上有三步操作(1.申请内存 2.调用构造方法初始化实例 3.把内存的首地址给single引用)在单线程的情况下指令2与3先执行哪个效果都一样,在多线程情况下,如果发送指令重排序执行步骤为1 3 2,在线程1执行了1 3后在执行2之前(也就是single这个引用他有了地址但是内存上的数据无效),此时线程2调用了这方法,在第一层if里判断时single非空不进入if直接return,就返回了一个不完整的对象。后续使用这个单例对象进行解引用,就会出现问题。要解决这个指令重排序带来的线程安全问题我们可以使用volatile这个关键字,它禁止了指令重排序,它既能够保证内存可见性,也能解决指令重排序问题,最终代码如下:

public class Single {
    private static volatile Single single;

    private Single(){}

    public static Single getInstance(){
        // 如果是第一次获取就创建
        synchronized (Single.class){
            if (single == null){
                single = new Single();
            }
        }
        
        return single;
    }
}

此时我们可以回顾一下:懒汉模式单例模式为什么要加锁?为什么有两个if? 为什么要加volatile关键字?

三、静态内部类

类在初次加载的时候,会初始化静态变量、静态代码块、静态方法但是不会加载静态内部类。静态内部类只有在使用的时候才会被加载,因此在使用静态内部类实现单例模式的时候,是满足懒加载的。而且这种实现方式,在获取单例时,是只获取不去修改的,所以他跟饿汉模式一样是天然线程安全的。他既有饿汉模式的优点,也有懒汉模式的优点。

public class Single {
    private Single() {}

    private static class SingleHolder {
        private static final Single single = new Single();
    }

    public static final Single getInstance() {
        return SingleHolder.single;
    }
}

四、枚举

枚举也是在第一次被调用的时候才会去加载,因此他也满足懒加载,并且他也是只获取单例对象,并不能修改,所以他也是线程安全

public enum  Single {
    SINGLE;
    public Single getInstance(){
        return SINGLE;
    }
}

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

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

相关文章

跟20%的同行去竞争80%的蓝海市场不香吗?

近年来,由于科技的发展等诸多因素,跨境电商行业有了长足的发展空间,不少人也有想要入行的打算。对于不是很了解这一行业的新手来说,如何选择合适的跨境电商市场与平台就显得至关重要。 一直以来,作为全球第四大电商市…

Android自定义View实现横向的双水波纹进度条

效果图:网上垂直的水波纹进度条很多,但横向的很少,将垂直的水波纹改为水平的还遇到了些麻烦,现在完善后发布出来,希望遇到的人少躺点坑。思路分析整体效果可分为三个,绘制圆角背景和圆角矩形,绘…

阅读HAL源码之重点总结

HAL封装中有如下特点(自己总结的): 特定外设要设置的参数组成一个结构体; 特定外设所有寄存器组成一个结构体; 地址基本都是通过宏来定义的,定义了各外设的起始地址,也就是对应寄存器结构体的地…

问答系统(QA)调研

引言 智能问答系统广泛用于回答人们以自然语言形式提出的问题,经典应用场景包括:智能语音交互、在线客服、知识获取、情感类聊天等。根据QA任务,可以将QA大致分为5大类,分别为: 文本问答(text-based QA&am…

使用Chemistry Development Kit (CDK) 来进行化学SMILES子结构匹配

摘要 SMILES是一种用于描述化合物结构的字符串表示法,其中子结构搜索是在大规模化合物数据库中查找特定的结构。然而,这种搜索方法存在一个误解,即将化合物的子结构视为一个独立的实体进行搜索,而忽略了它们在更大的化合物中的上…

码匠 × OpenAI :快速生成 SQL 语句,提升开发效率!

目录 使用 OpenAI 生成 SQL 码匠连接与集成 OpenAI 总结 关于码匠 在码匠中,编写 SQL 语句,并结合码匠一系列开箱即用的组件实现复杂的业务逻辑,是很常见的应用开发场景。然而,不同的数据库在 SQL 增删改查操作语法、类型字段和…

【C++进阶】一些小知识点

const限定符 用const给字面常量起个名字(标识符),这个标识符就称为标识符常量;因为标识符常量的声明和使用形式很像变量,所以也称常变量。声明方式: const int a 77; const float PI 3.14159f&#xff…

单核CPU, 1G内存,也能做JVM调优吗?

最近,笔者的技术群里有人问了一个有趣的技术话题:单核CPU, 1G内存的超低配机器,怎么做JVM调优?这实际上是两个问题。单核CPU的超低配机器,怎么充分利用CPU?单核CPU, 1G内存的超低配机器,怎么做J…

python学习之OpenCV-Python模块的部分应用示例(生成素描图和动漫图)

文章目录前言一、图片转灰度二、对图片进行二值化处理三、对图片去除噪点四、调整图片透明度五、生成素描滤镜效果图(方法结合应用)六、生成动漫卡通滤镜效果图(方法结合应用)总结前言 OpenCV 是一个图像和视频处理库&#xff0c…

由Geoscene Enterprise 2.1 升级至Geoscene Enterprise 3.1

文章目录一、升级前工作二、升级Geoscene portal三、升级Web Adaptor(针对portal门户)四、升级Server 站点(作为门户托管服务器)五、升级Web Adaptor(针对server)六、升级Data Store需求:由GeoS…

bestSync外网转内网操作

一.外网笔记本操作 设置管理员密码打开笔记本电脑,设置管理员密码(如果已经设置了的,请忽略该操作):左下角鼠标左键点击->控制面板->用户帐户和家庭安全->用户帐户->密码,密码设置完成后回到桌…

【Acwing 周赛复盘】第90场周赛复盘(2023.2.11)

【Acwing 周赛复盘】第90场周赛复盘(2023.2.11) 周赛复盘 ✍️ 本周个人排名:1488/2884 AC情况:1/3 这是博主参加的第五次周赛,这次做题的时候,感觉题目好难 😂 但是一听y总讲解,又…

ORA error集锦

1、oralce 数据客户端需要安装的问题 保存信息为: “无法连接到数据库,因为数据库客户端软件无法加载。确保已正确安装并配置数据库客户端软件” 从百度网盘下载,并安装win32 oracle client 安装包 2、ORA错误 “执行异常,ORA-00911: inval…

TCP四次挥手

TCP 四次挥手过程是怎样的? TCP 断开连接是通过四次挥手方式。 双方都可以主动断开连接,断开连接后主机中的「资源」将被释放,四次挥手的过程如下图: 客户端打算关闭连接,此时会发送一个 TCP 首部 FIN 标志位被置为 1…

Revit标注问题:尺寸界线长度和“快速尺寸定位标注”

一、 Revit中关于标注的问题 1.有时候,但我们建完模型要进行一定的标注,往往会出现这样的间题,如图1所示 按照正常的标注来说,通常它的标注的正确方式应该是这样的,尺寸界线长度应该是指向图元的,如图2所示…

【源码系列】Faster RCNN源码详解(一)——transform

系列文章目录 文章目录系列文章目录前言一、transform二、总结1.标准化2.缩放3.batch前言 Faster RCNN的源码整体可以分为7个模块,每个模块负责不同的功能。推荐B站up霹雳吧啦Wz讲解的Faster RCNN源码,已经很详细了,这里只是个人的一些理解总…

【Unity VR开发】结合VRTK4.0:创建滑块

语录: 只有经历地狱般的磨练,才能炼出创造天堂的力量。 前言: 滑块是一个非常简单的控件,它允许通过沿有限的驱动轴滑动 Interactable 来选择不同的值。我们将使用线性驱动器创建一个滑块控件,该控件允许我们根据与滑…

蓝桥杯刷题五

1.01背包问题这题就是01背包问题的模板题 回顾一下01背包 01就是这个东西选和不选01背包的表达式是f[i]max(f[i-v]w,f[i]);那么这题就可以直接做了 值得注意的是这里只用了一维数组 所以更新的时候要从后往前面更新#include <bits/stdc.h> using namespace std; const in…

【JDK8新特性之Stream流-Stream结果收集案例实操】

一.JDK8新特性之Stream流-Stream结果收集以及案例实操 二.Stream结果收集(collect函数)-实例实操 2.1 结果收集到集合中 /*** Stream将结果收集到集合中以及具体的实现 collect*/Testpublic void test01(){// 收集到List中 接口List<Integer> list Stream.of(1, 2, 3…

码住!新手容易上手的5个tiktok数据分析网站

当下短视频已经称霸了各大内容平台&#xff0c;越来越多的创作者进入到短视频赛道&#xff0c;为了更好地运营自己的内容平台&#xff0c;数据分析是必不可少的。很多人都入局了tiktok&#xff0c;对于商家或者博主红人来说&#xff0c;这是比较新平台&#xff0c;希望能在这个…