设计模式2:单例模式

单例模式

单例模式属于创建型模式 ,⼀个单例类在任何情况下都只存在⼀个实例,

构造⽅法必须是私有的、由⾃⼰创建⼀个静态变量存储实例,对外提供⼀

个静态公有⽅法获取实例。

双重检查锁(double check locking)

java 复制代码
public class LazySingleton {
    //volatile防止指令重排序
    private static volatile LazySingleton instance;
    private LazySingleton() {}
    public static LazySingleton getInstance() {
        //第一次检查,确保不必要的同步
        if (instance == null) {
            synchronized (LazySingleton.class) {
                //第二次检查,确保只有一个线程创建实例
                if (instance == null) {
                    instance = new LazySingleton();
                }
            }
        }
        return instance;
    }
}
  • 懒汉式,懒加载,使用的时候才创建对象

  • 双重检查指两次非空判断,锁指的synchronized加锁

    • 第一重判断,如果实例已经存在,无需进行同步操作
    • 第二重判断,有多个线程一起到达锁位置时,对锁竞争,其中一个线程获取锁后,第一次进入会判断实例为null,从而创建对象。完成后释放锁,其他线程获取锁后会被判定非空拦截,直接返回单例对象
  • synchronized同步代码块作用,防止有多个线程同时调用时,导致生成多个实例。有了同步块,每次只有1个线程能访问同步块内容。当第一个抢到锁线程创建实例后,之后的调用都不会进入同步块

  • volatile关键字作用:可见性、防止指令重排序。用new创建对象时,其实是三步操作,不是原子操作:

    • 1.在堆内存申请空间
    • 2.调用构造方法,初始化对象
    • 3.将引用变量指向堆内存空间

    为提高性能,处理器可能对代码执行顺序重新排序,如果运行顺序为1 3 2,当引用变量指向内存空间时,这个对象不为null,但是还没初始化,其他线程在调用getInstance方法使用时,会判断instance不为null,导致错误使用,出现异常

饿汉式

在类加载阶段就创建实例了

java 复制代码
public class EagerSingleton {
    private final static EagerSingleton instance = new EagerSingleton();

    private EagerSingleton() {}

    public static EagerSingleton getInstance() {
        return instance;
    }
}
  • 线程安全,没加锁,效率高
  • 缺点是类加载就初始化,浪费内存空间

有关创建型设计模式的内容就更新到这了,下一篇更新结构型设计模式的内容

相关推荐
程序猿小D41 分钟前
[附源码+数据库+毕业论文]基于Spring+MyBatis+MySQL+Maven+jsp实现的个人财务管理系统,推荐!
java·数据库·mysql·spring·毕业论文·ssm框架·个人财务管理系统
佛祖让我来巡山42 分钟前
【工厂和策略设计模式妙用】解决接口选择与多重if-else 问题
设计模式·策略模式·工厂模式
转转技术团队1 小时前
二奢仓店的静默打印代理实现
java·后端
钢铁男儿2 小时前
C# 接口(什么是接口)
java·数据库·c#
丶小鱼丶2 小时前
排序算法之【归并排序】
java·排序算法
上上迁2 小时前
分布式生成 ID 策略的演进和最佳实践,含springBoot 实现(Java版本)
java·spring boot·分布式
永日456702 小时前
学习日记-spring-day42-7.7
java·学习·spring
龙谷情Sinoam2 小时前
扩展若依@Excel注解,使其对字段的控制是否导出更加便捷
java
二十雨辰2 小时前
[尚庭公寓]07-Knife快速入门
java·开发语言·spring
掉鱼的猫3 小时前
Java MCP 实战:构建跨进程与远程的工具服务
java·openai·mcp