7.多线程之单例模式

单例模式

文章目录

  • 单例模式
  • [1. 什么是单例模式](#1. 什么是单例模式)
  • [2. 饿汉模式](#2. 饿汉模式)
  • [3. 懒汉模式](#3. 懒汉模式)
    • [3.1 单线程版:](#3.1 单线程版:)
    • [3.2 多线程版](#3.2 多线程版)

1. 什么是单例模式

单例模式是一种设计模式,常见的设计模式还有工厂模式、建造者模式等。

设计模式是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性、程序的重用性。我们使用这些设计模式,相当于是前辈们 在一些特殊的场景 下总结出的经验 ,我们直接使用即可,相当于站在巨人的肩膀上编程!!!

单例模式是针对一些特定的类,只允许创建一个实例 。为什么会有单例模式?因为引入单例模式后,在代码层面会有强制检查,不会允许创建出第二个实例,避免一些bug,人为检查是及其不可靠、不可控的。

单例模式有两种实现方式,分别是饿汉模式和懒汉模式,我们就这两种实现展开。

2. 饿汉模式

Java 复制代码
class Singleton{
    private static Singleton instance = new Singleton();
    public static Singleton getInstance() {
        return instance;
    }
    private Singleton() {}
}
public class Test {
    public static void main(String[] args) {
        Singleton sgt1 = Singleton.getInstance();
        Singleton sgt2 = Singleton.getInstance();
        System.out.println(sgt1 == sgt2);
    }
}    

运行结果~~

Java 复制代码
true

进程已结束,退出代码为 0

饿汉模式首先是线程安全的,对象在类加载时就会创建好,然后将构造方法设为私有,不允许再创建实例,最后通过公有方法将这个对象提供给外层调用。

这种模式为什么叫饿汉模式呢?其实也很简单,无论外层需不需要使用这个对象,在类加载时这个对象就已经创建了。而相比于饿汉模式,懒汉模式的做法则更加高效,在第一次实用类的时候,在进行创建。

3. 懒汉模式

3.1 单线程版:

Java 复制代码
class Singleton {
    private static Singleton instance = null;
    private Singleton() {}
    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
       }
        return instance;
   }
}

为什么说这样实现懒汉模式是单线程版本?因为这个版本的懒汉模式在并发环境下,会产生线程安全问题,即线程不安全。

线程安全问题发生在首次创建实例时. 如果在多个线程中同时调getInstance 方法, 就可能导致创建出多个实例。一旦实例已经创建好了, 后面再多线程环境调用 getInstance 就不再有线程安全问题了(不再修改instance 了),因此枷锁(synchronized)可以一定程度上改善这个问题。

这个版本的懒汉模式,在多线程环境下,一但有其他线程在第一次创建对象时穿插(OS系统调度)进行执行,就可能会创建出多个实例。

3.2 多线程版

Java 复制代码
class SingletonLaze{
    private volatile static SingletonLaze instance = null;
    public static SingletonLaze getInstance() {
        if(instance == null) {
            synchronized(SingletonLaze.class) {
                if (instance == null) {
                    instance = new SingletonLaze();
                }
            }
        }
        return instance;
    }
    private SingletonLaze() {}

在这个版本中对if判断进行枷锁,保证了创建对象时不会有其它线程穿插执行,在进一步给instance使用volatile修饰(保证可见性、顺序),从而保证了线程安全。我们进一步发现,只有在第一次进入时,才会需要创建对象,而如果每次都枷锁,就会降低程序的执行效率,所以我么在外部又嵌套了一层if。此处要注意的是这两层if虽然条件一样,但是含义却不一样。

锁竞争(枷锁解锁)的开销是比较大的,而懒汉模式只有在第一次才需要创建对象(产生锁竞争),后续就不在需要锁竞争了,也不必枷锁。

  1. 外层if :外层if是判断单例模式的对象是否创建出来,如果已经创建,则直接返回instance(instance被volatile修饰,不会产生内存可见性问题),不会产生锁竞争。
  2. 内层if:当线程第一次执行时会进入内层,竞争同一把锁,假设线程1拿到锁,就可以创建出对象,此时线程1执行完后,由于没内存可见性问题,其它线程就会发现instance已经不为空了,从而不会继续进入外层if竞争锁对象,降低了锁竞争的开销。

如果本篇文章对你有帮助,请点赞、评论、转发,你的支持是我创作的动力!!!

相关推荐
lssjzmn5 分钟前
Java并发容器ArrayBlockingQueue与LinkedBlockingQueue对比PK
java·消息队列
十五年专注C++开发8 分钟前
通信中间件 Fast DDS(二) :详细介绍
linux·c++·windows·中间件·fastdds
用户984089050872428 分钟前
Java基础之深拷贝浅拷贝-Integer
java
渣哥34 分钟前
99%的人忽略了!Java Integer缓存池原来暗藏玄机
java
小蒜学长40 分钟前
vue家教预约平台设计与实现(代码+数据库+LW)
java·数据库·vue.js·spring boot·后端
天天摸鱼的java工程师1 小时前
谈谈你对 Seata 的理解?8 年 Java 开发:从业务踩坑到源码级解析(附实战代码)
java·后端·面试
Emrys_1 小时前
基于 AOP 实现接口幂等性 —— 深入浅出实战指南
java
用户3721574261351 小时前
Java PPT转多种图片格式:打造高质量的文档转换服务
java
LSTM971 小时前
如何使用Java将PDF转换为Word
java
华仔啊1 小时前
SpringBoot+MySQL+Vue实现文件共享系统
java·前端·后端