Java多线程的单例设计模式 多种实现方法

news2025/1/11 6:56:56

目录

前言

饿汉式

懒汉式

Double_check

 volatile + double_check

Holder方式

 枚举


前言

        单例设计模式GOF23中设计模式中最常用的设计模式之一, 单例设计模式提供了多线程环境下的保证唯一实例性的解决方案, 虽然简单, 但是实现单例模式的方式多种多样, 因此需要从多个维度去评价: 

  • 线程安全
  • 性能
  • 懒加载

懒加载的好处是啥? 

提高页面加载速度:通过仅在需要时加载资源,懒加载可以减少初次加载时的资源量,使页面更快地呈现给用户。

节省带宽:只有当用户真正需要时才加载资源,这样可以节省不必要的带宽使用,特别是对于那些需要大量数据或高分辨率图片的应用。

提升用户体验:通过延迟加载,可以让用户更快地看到页面的主要内容,从而提高用户的满意度和体验。

减少服务器负载:懒加载可以减少一次性加载大量资源的需求,从而减少服务器的负载,优化服务器性能。

优化移动端性能:对于移动设备,网络连接通常较慢且不稳定,懒加载可以显著提高移动端的页面加载速度和性能。

节省系统资源:懒加载可以避免不必要的资源消耗,减少浏览器和设备的内存使用,尤其是在处理大量数据或复杂页面时。

饿汉式

public class TestMain {
    // 定义的时候直接初始化
    private static TestMain instance = new TestMain ();

    // 提供get方法
    public static TestMain getInstance() {
        return instance;
    }

    // 私有化构造方法
    private TestMain() {

    }
}

        在类加载过程中提到过, instance作为类变量, 在类的初始化的过程中, 会被收集到<cinit>()反方法中, 该方法每个类只会执行一次, 100%保证同步, 因此在使用instance的过程中, instance不可能会被实例化两次

        但是, 这也正是因为是在类初始化的时候就已经存在, 那么就表明, 这个instance实例在完成初始化之后, 并存储在内存空间之后, 并不会立即就会被使用到, 那么就会在内存中驻留一个时间间隔, 这个时间间隔带来的成本增加需要通过量化, 不能一概而论

        例如如果instance实例的内存消耗小, 那么饿汉式也未免不是一种良好的解决方案, 虽然他不支持懒加载.

        对于这个getInstance方法, 没有加任何锁, 因此没有其他的性能消耗, 速度快

懒汉式

        所谓懒汉, 就是我很懒, 我只有在使用类实例的时候才去创建, 可以避免类在初始化的时候就提前去创建.

代码如下: 

public final class Test {
    private static  Test instance = null;

    private Test() {
    }

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

        当前方法的getInstace, 每次进入这个方法之前都会获取这个类的Class对象, 然后加锁, 但是我每次调用getIsntace都需要拿到一次锁, 这样性能不高. 同一个时刻只能由一个线程能拿到这个instance实例. 

为了避免上述情况, 我们考虑降低这个锁的粒度: 

你似乎可以这样写: 

public final class Test {
    private static  Test instance = new Test ();

    private Test() {
    }

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

        但是这样写, 好像每次getInstance获取实例还是需要拿到锁, 还是会产生额外的锁竞争消耗 , 但是你仔细分析其实不难得知, 我除了第一次拿的时候需要初始化, 其他的时候, 都不需要初始化, 直接返回就行了, 于是你写出了这样的代码: 

public final class Test {
    private static  Test instance = new Test ();

    private Test() {
    }

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

但是第一次的时候会存在多线程风险, 也就是可能会存在多次给instance赋值的情况.... 

于是为了线程安全, 就给他加了锁, 如下: 

public final class Test {
    private static Test instance = new Test ();

    private Test() {
    }

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

}

 但是这样似乎并不安全, 如果有两个线程都进入了这个if判断中, 同时判断了这个instance为null都准备拿锁进行实例化, 假设是线程1拿到锁, 之后instance 被初始化, 然后被返回, 释放锁后, 线程2又走了一遍实例化的过程, 因此这个instance就被实例化了两次, 也就没有保障单列模式. 

于是就有了下面这种写法. 

Double_check

        不再在方法上加锁, 我们在方法内部使用锁块来进行加锁, 降低粒度. 如下: 

public final class Test {
    private static Test instance = null;

    private Test() {
    }

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

}

 volatile + double_check

public final class Test {
    private static Test instance = null;

     Object var1;

     Object var2;

    private Test() {
    }

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

    public static void setInstance(Test instance) {
        Test.instance = instance;
    }

    public Object getVar1() {
        return var1;
    }


    public Object getVar2() {
        return var2;
    }

}

但是同时这个方法还存在一个问题, 都知道java的内存模型中每个线程都有自己的工作内存,. 变量都是暂存在工作内存中, 对工作内存中修改变量, 如果没有的就去内公共内存中去读取, 那么档期使用这种方法更新的时候, 其实其他线程是看不到第一个线程对其的修改. 

除此之外, 上述的doublecheck也不一定安全, 看看上面的一个案例

        在没有volatile的情况下,编译器可能会将instance = new Test();这行代码拆分为多个操作:1) 分配内存给instance;2) 调用Test的构造函数来初始化对象;3) 将instance指向分配的内存。如果没有适当的同步措施,这些操作的执行顺序可能会被重排序,比如先执行步骤1和3,再执行步骤2。这样,在步骤2完成之前,其他线程就可能看到非空的instance引用,但此时对象可能还未被完全初始化,从而导致程序出错。 

代码如下: 

public final class Test {
    private volatile static Test instance = null;

     Object var1;

     Object var2;

    private Test() {
    }

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

    public static void setInstance(Test instance) {
        Test.instance = instance;
    }

    public Object getVar1() {
        return var1;
    }


    public Object getVar2() {
        return var2;
    }

}

Holder方式

在讲解java的jvm类加载机制的时候提到过: 

可以利用这个特性, 在需要实现单例模式的类中 添加一个静态内部类, 此时只要使用到了这个静态内部类里面的静态变量, 就会执行这个类的初始化, 但是后面无论如何 访问多少次, 就不再初始化了, 这就保证了单例模式的线程安全, 同时实现了懒加载, 重复获取也没有锁竞争. 

public final class Test {
    private Test() {}
    private static class Holder {
        private static final Test INSTANCE = new Test();
    }

    public static Test getInstance() {
        return Holder.INSTANCE;
    }

}

 枚举

        effect java力荐方式: 

在Java中,使用枚举(Enum)来实现单例模式是一种既简单又高效的方法。枚举类型在Java中是一种特殊的类,它默认就是单例的,并且线程安全。这是因为JVM保证了枚举实例的唯一性,并且在枚举的任何地方访问枚举实例时,都不需要同步

public enum SingletonEnum {  
    INSTANCE;  
  
    // 可以在这里添加方法  
    public void doSomething() {  
        System.out.println("Doing something...");  
    }  
  
    // 示例:添加私有字段和方法  
    private Singleton instance;  
  
    public void setData(Singleton instance) {  
        this.instance= instance;  
    }  
  
    public Singleton getData() {  
        return instance;  
    }  
  
}

其中: 

SingletonEnum instance1 = SingletonEnum.INSTANCE;  
SingletonEnum instance2 = SingletonEnum.INSTANCE;  

instance1和instance2是相同的实例.

当然这种方式不是懒加载的, 如果你想使用懒加载, 那么你可以使用静态内部类类似的加载机制: 

java的枚举类型和静态内部类一样, 只有在第一次被引用时会被加载和初始化,因此 EnumHolder 只有在调用 getInstance() 方法时才会被加载

public final class Test {
    private Test() {}

    private enum EnumHolder {
        INSTANCE;

        private Test test;

        EnumHolder() {
            test = new Test();
        }

        private Test getTest() {
            return test;
        }
    }

    // 调用该方法会主动使用 EnumHolder.INSTANCE, EnumHolder.INSTANCE

    public static Test getInstance() {
        return EnumHolder.INSTANCE.getTest();
    }

}

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

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

相关文章

【云原生】恰当运用kubernetes中三种探针,确保应用程序在Kubernetes集群中保持健康、可用和可靠

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

Unity热更——ILRuntime安装接入

一、ILRuntime相关地址 1、官网文档地址 2、GitHub上开源的ILRuntime项目工程 3、GitHub上的官方ILRuntime-Unity实例工程 4、官方视频教程-Unity中文课堂&#xff08;需付费&#xff09; 5、ILRuntime入门笔记-赵青青-博客园 6、ILRuntime 的学习-简书 二、ILRuntime…

Python | Leetcode Python题解之第324题摆动排序II

题目&#xff1a; 题解&#xff1a; def quickSelect(a: List[int], k: int) -> int:seed(datetime.datetime.now())shuffle(a)l, r 0, len(a) - 1while l < r:pivot a[l]i, j l, r 1while True:i 1while i < r and a[i] < pivot:i 1j - 1while j > l an…

自动生成数据:Navicat 16 让数据测试更高效

文章目录 前言一、Navicat是什么&#xff1f;二、Navicat 16 新功能1. 自动生成数据2. 改进的用户界面3. 云同步 三、 安装指南Windows 版安装macOS 版安装Linux 版安装 四、使用示例&#xff1a;自动生成数据1. 创建连接2. 选择表3. 打开数据生成器4. 设置数据规则5. 生成数据…

top命令实时监测Linux进程

top命令可以动态实时显示Linux进程信息&#xff0c;方便观察频繁换进换出的内存的进程变化。 top命令执行示例如下&#xff1a; 其中&#xff0c;第一行表示系统当前时间、系统的运行时间、登录的用户数目、系统的平均负载&#xff08;最近1分钟&#xff0c;最近5分钟&#xff…

springboot艺体培训机构业务管理系统--论文源码调试讲解

第2章 开发环境与技术 开发艺体培训机构业务管理系统需要搭建编程的环境&#xff0c;也需要通过调查&#xff0c;对各个相关技术进行分析&#xff0c;选取适合本系统开发的技术与工具。 2.1 MySQL数据库 MySQL是一种具有安全系数、安全系数、混合开发性、高效化等特征的轻量…

基于Raft算法的分布式KV数据库:五、剩余部分

github地址&#xff1a;https://github.com/1412771048/Raft CPPRaft系列-剩余部分 首先我们看下第五章的架构图&#xff0c;图中的主要部分我们在前几张讲解完毕了&#xff0c;目前还剩下clerk和k-v数据库&#xff0c;而本篇的重点在于补全版图&#xff0c;完成&#xff1a;…

SQL注入sqli-labs-master关卡一

本文环境搭建使用的是小皮&#xff0c;靶机压缩包&#xff1a;通过百度网盘分享的文件&#xff1a;sqli-labs-php7-master.zip 链接&#xff1a;https://pan.baidu.com/s/1xBfsi2lyrA1QgUWycRsHeQ?pwdqwer 提取码&#xff1a;qwer 下载解压至phpstudy的WWW目录下即可。 第一…

关于Redis的面试题

一、为什么要使用Redis 内存数据库,速度很快工作单线程worker,串行化,原子操作,IO线程是多线程的。避免上下文切换使用 IO模型,天生支撑高并发kv模型,v具有类型结构具有本地方法,计算数据移动二进制安全,value值最大为512MB二、Redis是多线程还是单线程 Redis在6.0版本…

(免费领源码)java#SSM#MYSQL私家车位共享APP 51842-计算机毕业设计项目选题推荐

目 录 摘要 1 绪论 1.1 课题的研究背景 1.2研究内容与研究目标 1.3ssm框架 1.4论文结构与章节安排 2 2 私家车位共享APP系统分析 2.1 可行性分析 2.2 系统流程分析 2.2.1 数据增加流程 2.2.2 数据修改流程 2.2.3数据删除流程 2.3 系统功能分析 2.3.1功能性分析 2…

计算机网络(TCP报文段首部格式中序号和确认号)

TCP首部格式中的序号和确认号并不总是同时出现。 TCP首部的序号和确认号是根据TCP通信的不同阶段和目的来决定的。在建立连接的过程中&#xff0c;序号用于标识发送数据的起始位置&#xff0c;而确认号用于表示接收方期望接收的下一个数据的起始位置。这两个字段在TCP通信的不同…

【vulhub靶场之rsync关】

一、使用nmap模块查看该ip地址有没有Rsync未授权访问漏洞 nmap -p 873 --script rsync-list-modules 加IP地址 查看到是有漏洞的模块的 二、使用rsync命令连接并读取文件 查看src目录里面的信息。 三、对系统中的敏感文件进行下载——/etc/passwd 执行命令&#xff1a; rsy…

【Python】Python中的循环语句

循环语句 导读一、基本概念1.1 循环语句的执行流程1.2 循环语句的分类 二、while语句三、for语句四、break与continue五、死循环六、循环中的else语句七、range()函数结语 导读 大家好&#xff0c;很高兴又和大家见面啦&#xff01;&#xff01;&#xff01; 在上一篇内容中我…

《Advanced RAG》-04-深度研究RAG技术Re-ranking

摘要 文章首先介绍了重新排序在RAG中的重要性&#xff0c;它允许对检索到的文档进行重新排序和过滤&#xff0c;以确保最相关的文档能够被优先考虑&#xff0c;从而提高RAG的效率和准确性。 接着&#xff0c;文章详细描述了两种主流的重新排序方法&#xff1a; 一种是使用重新排…

使用Jetbrains.Rider反编译Unity的DLL文件看源码

直接将dll文件的打开方式用Rider打开即可&#xff0c;打开BattleSeqGenertor.dll文件的效果如下&#xff1a;

Redis 的6种回收策略(淘汰策略)详解

Redis 的6种回收策略&#xff08;淘汰策略&#xff09;详解 1、Redis的六种淘汰策略1. volatile-lru2. volatile-ttl3. volatile-random4. allkeys-lru5. allkeys-random6. no-eviction 2、使用策略规则 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&am…

MyIP:强大且简单好用!

在这个数字化的时代&#xff0c;IP地址就像是我们的网络身份证。各位在日常的工作中&#xff0c;肯定会会遇到需要和 IP 地址相关的需求。 今天和大家聊一聊一个非常好用的开源 IP 工具项目 - MyIP。 简介 MyIP一个开源IP工具箱&#xff0c;提供了一系列的网络检测工具&…

适合双11入手的蓝牙耳机推荐?4款开放式耳机测评

2024年也确实快到大家购物疯狂买买买的双11日子了&#xff0c;我相信肯定也有人在购物车攒了一大堆商品就等着双11清空了。那肯定现在还有人在为双11的购物车放什么东西发愁吧&#xff0c;那对于我来说&#xff0c;双11的购物车应该也是要有蓝牙耳机的一席之地的。 因为毕竟在…

Python酷库之旅-第三方库Pandas(068)

目录 一、用法精讲 271、pandas.Series.dt.dayofyear属性 271-1、语法 271-2、参数 271-3、功能 271-4、返回值 271-5、说明 271-6、用法 271-6-1、数据准备 271-6-2、代码示例 271-6-3、结果输出 272、pandas.Series.dt.days_in_month属性 272-1、语法 272-2、参…

吴恩达老师机器学习作业-ex7(聚类)

导入库&#xff0c;读取数据&#xff0c;查看数据类型等进行分析&#xff0c;可视化数据 import matplotlib.pyplot as plt import numpy as np import scipy.io as sio#读取数据 path "./ex7data2.mat" data sio.loadmat(path) # print(type(data)) # print(data…