单例模式的设计与实现

文章目录

单例模式可能是我们在开发中用得最多的设计模式之一,但要在多线程环境下正确实现单例模式却不是那么简单。今天我们就来看看如何正确地实现线程安全的单例模式。

一、不安全的单例模式

在单线程环境下,实现单例模式很简单,但在多线程环境下就不安全了,可能会出现多个线程同时创建实例的情况,这就违背了单例模式的初衷。

java 复制代码
// 这种实现在多线程环境下是不安全的
public class Singleton {
    private static Singleton instance;
    
    private Singleton () {}
    
    public static Singleton getInstance() {
        if (instance == null) {
            // 如果多个线程同时执行到这里,就可能创建多个实例
            instance = new Singleton ();
        }
        return instance;
    }
}

如果线程A和线程B同时调用getInstance方法,都发现instance为null,那么它们可能会同时创建实例,这就破坏了单例的约束。

二、线程安全的单例模式

1. 同步方法

最直接的解决方案就是给getInstance方法加上synchronized关键字:

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

这种方式确实能保证线程安全,但是有个明显的性能问题:每次调用getInstance都需要获取锁,即使实例已经创建好了。在高并发场景下,这会成为一个性能瓶颈。

2. 静态内部类

java 复制代码
public class StaticInnerClassSingleton {
    
    private StaticInnerClassSingleton() {}
    
    private static class SingletonHolder {
        private static final StaticInnerClassSingleton INSTANCE = new StaticInnerClassSingleton();
    }
    
    public static StaticInnerClassSingleton getInstance() {
        return SingletonHolder.INSTANCE;
    }
}

这种实现利用了JVM的类加载机制来保证线程安全。静态内部类SingletonHolder只有在被引用时才会被加载,而类的加载过程是线程安全的,JVM会保证一个类只被加载一次。

但是普通的单例实现可能会被反射攻击破坏:

java 复制代码
// 反射攻击示例
Constructor<StaticInnerClassSingleton> constructor = 
    StaticInnerClassSingleton.class.getDeclaredConstructor();
constructor.setAccessible(true);
StaticInnerClassSingleton instance1 = constructor.newInstance();
StaticInnerClassSingleton instance2 = StaticInnerClassSingleton.getInstance();
// instance1 和 instance2 是不同的对象

3. 枚举:最安全的实现方式

在《Effective Java》中推荐了使用枚举来实现单例模式:

java 复制代码
public enum EnumSingleton {
    INSTANCE;
    
    public void doSomething() {
        // 业务方法
    }
}

// 使用方式
EnumSingleton.INSTANCE.doSomething();

枚举实现单例有几个独特的优势

  1. 枚举的实例创建是线程安全的,JVM会保证枚举实例只被创建一次。

  2. 但是枚举天然防止反射攻击,因为JVM不允许通过反射创建枚举实例。

相关推荐
Boop_wu2 分钟前
简单介绍 JSON
java·开发语言
知识即是力量ol8 分钟前
初识 Kafka(一):分布式流平台的定义、核心优势与架构全景
java·分布式·kafka·消息队列
爱吃生蚝的于勒12 分钟前
【Linux】线程概念(一)
java·linux·运维·服务器·开发语言·数据结构·vim
kong790692813 分钟前
Nginx性能优化
java·nginx·性能优化
Pluchon14 分钟前
硅基计划4.0 算法 简单模拟实现位图&布隆过滤器
java·大数据·开发语言·数据结构·算法·哈希算法
我命由我1234514 分钟前
Java 泛型 - Java 泛型通配符(上界通配符、下界通配符、无界通配符、PECS 原则)
java·开发语言·后端·java-ee·intellij-idea·idea·intellij idea
Seven9715 分钟前
AQS深度探索:以ReentrantLock看Java并发编程的高效实现
java
4311媒体网23 分钟前
C语言操作符全解析 C语言操作符详解
java·c语言·jvm
淡忘_cx24 分钟前
使用Jenkins自动化部署spring-java项目+宝塔重启项目命令(2.528.2版本)
java·自动化·jenkins
毕设源码-钟学长28 分钟前
【开题答辩全过程】以 基于SSM的孤儿救助信息管理系统设计与实现为例,包含答辩的问题和答案
java