Java 三种方式实现单例模式

单例模式是一种常见的设计模式,它用于确保一个类只有一个实例,并提供全局访问点。在许多情况下,我们可能需要确保一个类只能创建一个对象,并且能够全局访问这个对象。单例模式正是为了满足这种需求而生。在Java中,实现单例模式的方式多种多样,包括懒汉式、饿汉式、双重检查锁定、静态内部类等。本文将总结和讨论这些方式的实现、优缺点以及线程安全性,并提供一种使用枚举类来实现单例模式的方式。通过本文的介绍,读者将能够深入了解单例模式的各种实现方式,以及在不同情况下如何选择合适的单例模式实现方法。

1. 双重检查锁

Java 单例模式中的双重检查锁(Double-Checked Locking)是一种常见的实现方式,旨在确保线程安全的同时提高性能。

csharp 复制代码
class Singleton {
    private static volatile Singleton singleton;

    private Singleton() {}

    public Singleton get() {
        if (singleton == null) {
            synchronized (this) {
                if (singleton == null) {
                    return new Singleton();
                }
            }
        }

        return singleton;
    }
}

针对双重检查锁的详细解读:

  • volatile 关键词:避免JVM的指令重排。
  • synchronized 关键词:避免并发 new 对象,导致对象被覆盖。
  • singleton == null 外层校验:单例对象创建出来后就不需要进入synchronized的锁逻辑了,提高读取效率。
  • singleton == null 内层校验:避免进入synchronized关键词后对象的重复创建。

双重检查锁的问题:使用volatile关键字能保证执行顺序,但是在一定程度上会影响执行效率。

2. 静态内部类

双重检查锁是一种有效且常用的单例模式实现方式,但在某些情况下可能会遇到无序写问题。为确保线程安全和避免潜在问题,建议使用静态内部类的方式来实现单例模式。这种方式不仅简洁高效,还能确保在任何情况下都能正确地创建和访问单例实例。

csharp 复制代码
public class Singleton {
    private Singleton() {}

    private static class SingletonHolder {
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

优点:

  1. 线程安全:静态内部类的初始化是线程安全的,由 JVM 保证。
  2. 延迟加载 :只有在第一次访问 getInstance() 方法时才会加载内部类并创建实例。
  3. 简洁高效 :无需显式使用 synchronized 关键字,减少了代码复杂度。

3. 枚举

在Java中,我们也可以使用枚举类来实现单例模式。由于枚举类的特性保证了只有一个枚举常量对象,因此可以通过枚举类来实现单例模式,且代码简洁且具有线程安全的特性。

csharp 复制代码
public enum Singleton {
    INSTANCE;  // 唯一的枚举实例

    // 添加其他属性或方法
    public void doSomething() {
        // 具体操作
    }
}

使用枚举类实现单例模式的好处是简洁、安全且可靠。不过,在我们一般的业务系统中,枚举类往往都是由自己的语义的,用枚举类实现单例模式并不常见。

相关推荐
java干货1 分钟前
虚拟线程与消息队列:Spring Boot 3.5 中异步架构的演进与选择
spring boot·后端·架构
一只叫煤球的猫6 分钟前
MySQL 8.0 SQL优化黑科技,面试官都不一定知道!
后端·sql·mysql
SoFlu软件机器人8 分钟前
智能生成完整 Java 后端架构,告别手动编写 ControllerServiceDao
java·开发语言·架构
写bug写bug1 小时前
如何正确地对接口进行防御式编程
java·后端·代码规范
Cyanto1 小时前
Java并发编程面试题
java·开发语言·面试
不超限1 小时前
Asp.net core 使用EntityFrame Work
后端·asp.net
在未来等你1 小时前
互联网大厂Java求职面试:AI大模型与云原生技术的深度融合
java·云原生·kubernetes·生成式ai·向量数据库·ai大模型·面试场景
豌豆花下猫1 小时前
Python 潮流周刊#105:Dify突破10万星、2025全栈开发的最佳实践
后端·python·ai
sss191s2 小时前
Java 集合面试题从数据结构到 HashMap 源码剖析详解及常见考点梳理
java·开发语言·数据结构