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三大集合:List、Set、Map
java·开发语言
存在的五月雨几秒前
Spring Security认证流程
java·开发语言·mysql
树码小子1 分钟前
综合练习:验证码案例(1)总体设计
java·开发语言·spring
一嘴一个橘子6 分钟前
idea 执行 Maven 的 `clean`、`install`、`package` 等命令报错
java
Sylvia-girl7 分钟前
线程通讯~
java
Victor35610 分钟前
MongoDB(2)MongoDB与传统关系型数据库的主要区别是什么?
后端
JaguarJack11 分钟前
PHP 应用遭遇 DDoS 攻击时会发生什么 从入门到进阶的防护指南
后端·php·服务端
BD_Marathon11 分钟前
设计模式——合成复用原则
设计模式·合成复用原则
BingoGo11 分钟前
PHP 应用遭遇 DDoS 攻击时会发生什么 从入门到进阶的防护指南
后端
Victor35612 分钟前
MongoDB(3)什么是文档(Document)?
后端