最常考的设计模式之一---单例模式

软件开发中有很多常见的 "问题场景". 针对这些问题场景, 大佬们总结出了一些固定的套路 ,这些套路就被称为设计模式

而我们今天要介绍的就是设计模式中的单例模式


单例模式的定义

单例模式是一种常见的设计模式,它确保一个类只有一个实例,并提供一个全局访问点。这在需要控制资源访问、限制对象数量、或者跨系统访问时非常有用。

单例模式具体的实现方式, 分成 "饿汉" 和**"懒汉"** 两种.

懒汉式单例 :在第一次被调用时创建实例

示例代码如下:

java 复制代码
public class SingleTon {

    //我们要创建的实例instance,因为全局只会存在一个,所以要加上static
    private static SingleTon instance;

    //构造方法要手动设成private的,不然外部可以随意创建实例,违背了一个实例的初衷
    private SingleTon(){}

    //获取instance的方法,如果instance为null就创建一个,所以也要加上static
    public static SingleTon getInstance(){
        if(instance==null){
            instance=new SingleTon();
        }
        return instance;
    }
    //最后返回instance
}

饿汉式单例 :在类加载时就创建实例

示例代码如下:

java 复制代码
public class SingleTon2 {

    我们要创建的实例instance,饿汉模式直接创建
    private static final SingleTon2 instance=new SingleTon2();

    private SingleTon2(){};

    //因为已经创建了,直接返回就行了
    public static SingleTon2 getInstance(){
        return instance;
    }

}

单例模式的安全隐患

上面的懒汉模式的代码在单线程中是没有问题的,但在多线程中就有安全隐患了

首次创建实例 时. 如果在多个线程 中同时调用 getInstance 方法 , 就可能导致 创建出多个实例

加上synchronized可以改善这里的线程安全问题.

java 复制代码
package Singleton;

public class SingleTon {

    private static SingleTon instance;

    private SingleTon(){}

    //加上synchronized关键字
    public static synchronized SingleTon getInstance(){
        if(instance==null){
            instance=new SingleTon();
        }
        return instance;
    }

}

不清楚synchronized 的可以参考Java中的监视器锁 (synchronized 关键字)-CSDN博客

当我们加上锁后,线程确实安全了,但我们每次获取instance实例 时,即使用getInstance()方法 时都要加锁 ,明明只有首次创建实例才需要加锁,后面都不需要加锁了,这就造成了很多多余的开销

所以我们改成getInstance()方法内部判断要不要加锁

代码如下:

java 复制代码
public class SingleTon {

    //加上volatile关键字可以让instance每次使用时强制刷新,避免数据没有及时更新
    private static volatile SingleTon instance;

    private SingleTon(){}

    public static SingleTon getInstance(){

        //首先判断instance是不是null,如果是就进入方法
        if(instance==null){

            //注意这里,因为前面的if是没有加锁的,所以可能同时有多个线程进入方法
            synchronized (SingleTon.class){

                //进入锁中,此时因为前面可能有多个线程进入if语句,所以再判断一下
                if(instance==null){
                    instance=new SingleTon();
                }

            }

        }
        return instance;
    }
}

到这里,关于单例模式的讲解就结束了~

相关推荐
It's now2 小时前
Spring AI 基础开发流程
java·人工智能·后端·spring
cxh_陈2 小时前
线程的状态,以及和锁有什么关系
java·线程·线程的状态·线程和锁
计算机毕设VX:Fegn08952 小时前
计算机毕业设计|基于springboot + vue图书商城系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端·课程设计
R.lin2 小时前
Java 8日期时间API完全指南
java·开发语言·python
毕设源码-赖学姐2 小时前
【开题答辩全过程】以 高校教学质量监控平台为例,包含答辩的问题和答案
java·eclipse
高山上有一只小老虎2 小时前
翻之矩阵中的行
java·算法
火钳游侠3 小时前
java单行注释,多行注释,文档注释
java·开发语言
code bean3 小时前
【CMake】为什么需要清理 CMake 缓存文件?深入理解 CMake 生成器切换机制
java·spring·缓存
selt7913 小时前
Redisson之RedissonLock源码完全解析
android·java·javascript
RestCloud3 小时前
智能制造的底层基建:iPaaS 如何统一 ERP、MES 与 WMS 的数据流
java·wms·erp·数据传输·ipaas·mes·集成平台