JAVA多线程中的单例模式

一.什么是单例模式?

在 Java 里,单例模式指的是一个类仅有一个实例,并且提供一个全局访问点**(公有get方法)来获取该实例**。在多线程环境下实现单例模式时,需要保证线程安全,避免多个线程同时创建多个实例。下面介绍几种常见的多线程单例模式实现方式

二.饿汉式

在类加载时就创建了实例

1.代码示例

java 复制代码
class Singleton {
    //在类加载时直接创建实例
    //static 修饰引用变量,意味着这个变量属于类,而不是实例
    //确保类只有一个实例。这里用 static 让 instance 属于类,只有一份。
    private static Singleton instance = new Singleton();
    //提供公有的get方法来让外部获取唯一实例
    public static Singleton getInstance() {
        return instance;
    }

    // 把构造方法设为 private。此时在类的外面,就无法继续 new 实例了.
    private Singleton() {
    }
}
 class Demo20 {
    public static void main(String[] args) {
        // 如果代码这么写,就违背初衷,就应该禁止这个类在类外部被 new.
        // Singleton instance2 = new Singleton();
        Singleton instance = Singleton.getInstance();
    }
}

2.以上代码是线程安全的------为什么?

java 复制代码
public static Singleton getInstance() {
        return instance;
    }

因为上述代码中这段操作只涉及到"读取数据",不涉及到修改

三.懒汉式

不是在类加载时就创建了实例

先来看一段代码

1.代码示例

java 复制代码
class SingletonLazy {
    private static SingletonLazy instance = null;

    public static SingletonLazy getInstance() {
        if (instance == null) {
            instance = new SingletonLazy();
        }
        return instance;
    }
    // 把构造方法设为 private。此时在类外部,就无法继续 new 实例了.
    private SingletonLazy() {}
}

 class Demo2 {
    public static void main(String[] args) {
        SingletonLazy instance = SingletonLazy.getInstance();
        SingletonLazy instance2 = SingletonLazy.getInstance();
    }
}

2.解释

这种方式在首次调用getInstance( )的时候才会创建实例,后续再调用getInstance( ),则直接返回实例(不new)。

3.但在多线程环境下不安全--为什么?

java 复制代码
 if (instance == null) {
            instance = new SingletonLazy();
        }
        return instance;

在上述代码中这段操作既涉及到了读取,也涉及到了修改(将SingletonLazy对象赋值给引用变量instance)

假设有两个线程 t1,t2进行下方操作

java 复制代码
SingletonLazy instance = SingletonLazy.getInstance();

导致实例被创建出多份

4.如何修改?

加锁

4.1 synchronized修饰此段代码块

java 复制代码
class SingletonLazy {
    private static SingletonLazy instance = null;

    public static SingletonLazy getInstance() {
        //这里
        synchronized (Singleton.class){
             if(instance == null) {
                instance = new SingletonLazy();
            }
        }
        return instance;
    }
   
    private SingletonLazy() {}
}

4.2 synchronized修饰getInstance( )方法

java 复制代码
class SingletonLazy {
    private static SingletonLazy instance = null;
    //这里
    public static synchronized SingletonLazy getInstance() {
      
             if(instance == null) {
                instance = new SingletonLazy();
            }

        return instance;
    }

    private SingletonLazy() {}
}

5.有个小的新问题

线程不安全并不是一直都不安全。只是在多线程(多个线程)环境下,首轮调用getInstance( )方法才会不安全。一旦实例创建好,多个线程同时调用getInstance( )方法也并不会触发线程不安全问题。

总:会带来一定的性能开销

6.如何解决?

怎样既保持线程安全,也减少同步带来的性能开销

6.1 解决方法

双重检查锁定

java 复制代码
public static SingletonLazy getInstance() {
        //加上判定条件
        if (instance==null){
            //在此处加锁
            synchronized (Singleton.class){
                if(instance == null) {
                    instance = new SingletonLazy();
                }
            }
        }

        return instance;
}

 

6.2 新问题 (这里可以不看)

6.2.1 解决方法(这里看)

使用 volatile 关键字保证可见性

java 复制代码
 private static volatile SingletonLazy instance = null;
相关推荐
云烟成雨TD1 小时前
Spring AI Alibaba 1.x 系列【69】Token 用量统计
java·人工智能·spring
JAVA9651 小时前
JAVA面试-并发篇 03-使用synchronized doublecheck实现单例有什么坑
java·单例模式·面试
在繁华处1 小时前
Java从零到熟练(四):面向对象基础
java·开发语言
Unbelievabletobe1 小时前
解决了股票api接口盘后数据更新慢的问题
大数据·开发语言·python
不会C语言的男孩2 小时前
C++ Primer 第2章:变量和基本类型
开发语言·c++
小江的记录本3 小时前
【JVM虚拟机】堆内存分代模型:年轻代(Eden+Survivor)、老年代、元空间Metaspace(附《思维导图》+《面试高频考点清单》)
java·前端·jvm·后端·python·spring·面试
在繁华处3 小时前
Java从零到熟练(三):流程控制
java·开发语言·python
唐青枫3 小时前
Java Optional 实战指南:优雅处理空值与链式转换
java
一起学开源3 小时前
一文读懂 ReAct 范式:让 AI Agent 真正学会“思考+行动“
java·javascript·react.js·ecmascript·react·alibaba·智能体开发
云泽8084 小时前
C++ 可调用对象通关指南:深度解析 Lambda 表达式、function 包装器与 bind 绑定器
开发语言·c++·算法