两种单例模式(保证线程安全)

开始前,球球各位读者给个三连吧,有错误感谢指出,谢谢

单例模式也叫单个实例,也就是这个类只有且只能有一个实例对象,这样一个类就叫做“单例”;单例模式有很多种,这里只介绍“饿汉模式”和“懒汉模式”两种;

饿汉模式

唯一实例创建的时机非常早

//饿汉模式
class Singleton{
    //私有静态的实例对象,外部无法获取,随着类的记载而加载
    //类对象属性,保证只有一个
    private static Singleton singleton=new Singleton();
    //共有的静态方法,外界能够直接直接通过类名访问,获取该单例对象
    public static Singleton getSingleton(){
        return singleton;
    }
    //私有方法,确保外部无法创建该对象的实例
    private Singleton(){}
}
public class demo5 {
    public static void main(String[] args) {
        Singleton s1= Singleton.getSingleton();
        Singleton s2= Singleton.getSingleton();
        //s1,s2指向的是同一个类对象
        System.out.println(s1==s2);//true
        Singleton s3=new Singleton();//报错,外部无法再new一个实例对象
    }
}

懒汉模式

只有当该类第一次被实例化的才实例化,如果该类没有被实例化就不实例化,且要确保后续无法再次实例化,确保只有一个实例化对象;

//"懒汉模式"
class SingletonLazy{
    //首先实例对象要设置为空
    private static SingletonLazy singletonLazy=null;
    //共有的获取该类唯一实例对象的方法
    public static SingletonLazy getSingletonLazy(){
        if(singletonLazy==null){
            singletonLazy=new SingletonLazy();
        }
        return singletonLazy;
    }
    //私有的构造方法,外部无法创建该类的实例化对象
    private SingletonLazy(){}
}
public class demo6 {
    public static void main(String[] args) {
        SingletonLazy s1=SingletonLazy.getSingletonLazy();
        SingletonLazy s2=SingletonLazy.getSingletonLazy();
        //s1,s2指向的是同一个实例化对象;
        System.out.println(s1==s2);//true
        //无法创建实例化对象
        SingletonLazy s3=new SingletonLazy();
    }
}

两种模式的最大的区别就是唯一对象创建的实际不同,饿汉模式会在第一个时间创建,只要该类加载内存,唯一实例化对象就随着被创建,而懒汉模式只有当被第一次调用需要实例化对象是才会实例化唯一对象,如果不调用,就不创建;

如果程序中包含多个单例类,使用饿汉模式,刚开始就会扎堆创建很多个单例对象,可能会使得程序启动变慢,如果是懒汉模式,只有当被调用时被会实例化对象,实际是分散的,不容易感觉卡顿;

单例模式一般运用在类只需要一个实例化对象的时候,或者需要避免该类被实例化第二个对象;举个例子:你写的服务器,要从硬盘上加载100G的数据到内存中,肯定要写成一个类,封装上述加载操作,并且写一些获取/处理数据的逻辑方法,这样的类,就因该是单例的,一个实例化就要管理100g的数据,多个实例,就要加载N*100G的数据,这是没有必要,机器也是吃不消的;

 

单例模式在多线程如何保证线程安全

饿汉模式在多线程是安全的,但是懒汉模式在多线程是不安全的;

饿汉模式是安全的因为在线程还没有创建之前,唯一实例化对象就随着类的加载而创建了,所以后续无论是多少个线程的单例对象指向的都是同一个实例化对象;

懒汉模式之所以在多线程是不安全,是因为在该模式下的实例化对象是在程序运行中被创建的,这其中就有可能多个线程同时实例化对象,不安全那就要加锁,但是加锁在哪里又是一个需要考虑的问题,结合下面两个线程同时实例化对象的可能过程一起来了解一下吧

第一种情况:锁在new对象的时候

a3764114f9f844e99dbb17b427a0f315.png

第二种情况:锁在判断对象是否为空的时候

0d4a5cc1fb2840d990dcc04877ee1e07.png

第二种情况的优化

第二种情况无疑是可以在一定程度上确保线程安全的,但是针对第二种情况我们还可以再做优化,由于以上例子只是在只有两个线程访问时确保线程安全,如果有多个线程访问,这时候后续的每一个线程在获取实例化对象的,还是每次都要加锁然后再去判断对象是否为空,加锁的开销也是很大的,是一定程度上会拖慢程序的运行的,这使得加锁在一定程度上与低效挂钩,所以第二种情况的优化就是在进行一个判断singletonLazy是否为空,代码如下:

public static SingletonLazy getSingletonLazy(){
       if(singletonLazy==null){
           synchronized (singletonLazy){
               if(singletonLazy==null){
                   singletonLazy=new SingletonLazy();
               }
           }
       }
        return singletonLazy;
    }

还有一点补充,虽然这一点不会100%出现,就是编辑器对代码进行了优化,出现内存可见性的情况,即singletonLazy存在寄存器里,当线程创建了该类的唯一实例化对象时,singletonLazy并没有修改指向该对象,所以可以在上volatile关键字;

以下时懒汉模式在多线程依然保持安全完整代码:

//"懒汉模式"
class SingletonLazy{
    //首先实例对象要设置为空
    private static volatile SingletonLazy singletonLazy=null;
    //共有的获取该类唯一实例对象的方法
    public static SingletonLazy getSingletonLazy(){
       if(singletonLazy==null){
           synchronized (singletonLazy){
               if(singletonLazy==null){
                   singletonLazy=new SingletonLazy();
               }
           }
       }
        return singletonLazy;
    }
    //私有的构造方法,外部无法创建该类的实例化对象
    private SingletonLazy(){}
}

 

 

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

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

相关文章

抖音商城618好物节消费数据报告发布,带货成交额同比增长300%

6月21日,“抖音商城618好物节”消费数据报告发布,呈现618期间平台全域经营情况及大众消费趋势。 今年618大促活动中,抖音电商投入流量资源和消费券,鼓励商家、达人双向经营货架场景和内容场景,不断激活消费市场。 报…

202406最新manjaro安装sogou输入法解决方案(采用aur本地package+sogou deb包解决方案)

本地执行安装方法 1.拉取源码 git clone https://gitee.com/liushuai05/fcitx-sogoupinyin.git cd fcitx-sogoupinyin 2.获取sogou下载地址并替换到源码中 - 下载地址:https://pinyin.sogou.com/linux/ - 点击立即下载->x86_64->下载,然后右键复…

超级管道,品质非凡——钢塑复合管

钢塑复合管,是一种新型的复合管材,又叫涂塑钢管,涂塑钢管有内涂塑钢管,外涂塑钢管,内外涂塑钢管,外镀锌内涂塑钢管,外3pe防腐内涂塑钢管等。 它结合了钢管和塑料管的优点,具有高强度…

Python学习路线

Python学习路线 领取资料 一、Python基础知识 Python入门:了解Python的安装方法、如何运行Python程序以及交互模式的使用,同时学习注释的添加方法。 数据类型:掌握Python中的各种数据类型,包括数字、布尔值、字符串、列表、元…

Golang笔记:使用serial包进行串口通讯

文章目录 目的使用入门总结 目的 串口是非常常用的一种电脑与设备交互的接口。这篇文章将介绍golang中相关功能的使用。 本文使用的包为 :go.bug.st/serial https://pkg.go.dev/go.bug.st/serial https://github.com/bugst/go-serial 另外还有一些常见的包如&…

Junit单元测试

就是针对最小的功能单元,编写测试代码对其进行正确性测试 Junit单元测试框架 public class StringUtil {public static void printNumber(String name) {if(name null){System.out.println(0);return; //停掉方法}System.out.println("名字长度是:" name.length(…

基于支持向量机的垃圾邮件分类,使用SVM+flask+vue

sms-classify 基于支持向量机的垃圾邮件分类,使用SVMflaskvue 数据集和源码地址 数据集 SMS Spam Collection Data Set 来源于 UCI。样例被分为非垃圾邮件(86.6%)和垃圾邮件(13.4%),数据格式如下&#xff…

Why RAG is slower than LLM?

I used RAG with LLAMA3 for AI bot. I find RAG with chromadb is much slower than call LLM itself. Following the test result, with just one simple web page about 1000 words, it takes more than 2 seconds for retrieving: 我使用RAG(可能是指某种特定的…

高考志愿填报,二个准备三个重点四个原则

对于高考生而言,高考完毕并不是可以轻松地开始,接下来需要研究怎么报考的问题。如何在理想和现实中取得平衡?如何根据就业和专业的前景做合适的安排,对于还处于青少年阶段的高考生们来说不是容易的事情,要掌握哪些技巧…

第2章 Android应用的界面编程

🌈个人主页:小新_- 🎈个人座右铭:“成功者不是从不失败的人,而是从不放弃的人!”🎈 🎁欢迎各位→点赞👍 收藏⭐️ 留言📝 🏆所属专栏&#xff1…

后端路线指导(4):后端春招秋招经验分享

后端春招&秋招经验分享 春招(暑期实习) /秋招是应届生非常重要的应聘时间,每一个想就业的同学一定要有所了解! 本篇内容,老白将与大家分享暑期实习和秋招如何应对招聘的个人经验,希望每个同学看完都能有所收获! 首先说明一下老白对于面试核心竞争力的…

Java比较运算符

关系运算符和比较运算符适用于条件判断类型。 相当于布尔值,只有True和False两个 符号 说明ab,判断a的值是否等于b的值,条件成立为true,不成立为false ! a!b,判断a和b的值是否不相等,条件成立为true,不成立为false > …

spring注解驱动系列-- spring容器创建原理

从源码开始探究spring容器的创建原理,下面是源码总步骤 Override public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// Prepare this context for refreshing.prepareRefresh();// Tell the subc…

Shell编程规范与变量-01

一、Shell脚本概述 在一些复杂的 Linux 维护工作中,大量重复性的输入和交互操作不仅费时费力,而且容易出错,而编写一个恰到好处的 Shell 脚本程序,可以批量处理、自动化地完成一系列维护任务,大大减轻管理员的负担。 1…

【数据结构与算法】线索二叉树 详解

为什么可在不增加指针域的情况下,对二叉树进行线索化? 不增加指针域:因为可以利用n1个空链域。 在线索二叉树中,为每个节点添加两个标志位,分别表示左指针和右指针是普通的孩子指针还是线索(前驱或后继&a…

WPS没保存关闭了怎么恢复数据?4个方法(更新版)

想象一下,你正在用WPS奋笔疾书,灵感如泉水般涌出,突然间,电脑却跟你开了个玩笑——啪地一下,文档未保存就关闭了!是不是感觉像是被泼了一盆冷水,所有的热情瞬间熄灭?别急&#xff0c…

Vue78-缓存路由组件

一、需求 路由切走的时候&#xff0c;组件会被销毁&#xff0c;路由切回来&#xff0c;组件被挂载&#xff01; 需要&#xff1a;路由切走的时候&#xff0c;组件不会被销毁。 二、代码实现 若是不加include属性&#xff0c;则在<router-view>里面展示的路由&#xff0c…

【MySQL进阶之路 | 高级篇】InnoDB存储结构(页的内部结构)

1. 数据库的存储结构 : 页 索引结构给我们提供了高效的索引方式&#xff0c;不过索引信息以及数据记录都是保存在文件上的.确切说是存储在页结构中.另一方面&#xff0c;索引是在存储引擎中实现的&#xff0c;MySQL服务器上的存储引擎负责对表中数据的读取和写入操作.不同的存…

MQTTfx连接阿里云(详细版)

1、介绍 作为物联网开放平台&#xff0c;阿里云可谓是吸引大多数嵌入式爱好者的平台。物联网MQTT协议火热的今天&#xff0c;你使用过阿里云吗&#xff1f;本篇文章带你接触阿里云&#xff0c;实现MQTT通信。 我们在测试MQTT之前先了解下什么是MQTT协议。大家都知道它是一种发…

面向对象修炼手册(一)(类与对象)(Java宝典)

&#x1f308; 个人主页&#xff1a;十二月的猫-CSDN博客 &#x1f525; 系列专栏&#xff1a; &#x1f3c0;面向对象修炼手册 &#x1f4aa;&#x1f3fb; 十二月的寒冬阻挡不了春天的脚步&#xff0c;十二点的黑夜遮蔽不住黎明的曙光 目录 思想 代理和团体 类 1 基本概…