java单例模式的正确写法

本文详细介绍了单例模式的六种实现方式,包括懒汉式(线程不安全)、懒汉式(线程安全)、双重检验锁、双重检验锁(volatile)、饿汉式及静态内部类等,并探讨了每种方式的特点及适用场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、懒汉式(线程不安全)

 

public class Singleton {
    private static Singleton instance;
    private Singleton (){}

    public static Singleton getInstance() {
     if (instance == null) {
         instance = new Singleton();
     }
     return instance;
    }
}

 

介绍:线程不安全,在多线程情况下容易创建多个实例。

 

 

二、懒汉式(线程安全)

 

public static synchronized Singleton getInstance() {
    if (instance == null) {
        instance = new Singleton();
    }
    return instance;
}

  

介绍:虽然线程安全,但是不够高效。

 

 

三、双重检验锁

 

public static Singleton getSingleton() {
    if (instance == null) {                         //Single Checked
        synchronized (Singleton.class) {
            if (instance == null) {                 //Double Checked
                instance = new Singleton();
            }
        }
    }
    return instance ;
}

 

介绍双重检验锁机制比懒汉式(线程安全)要高效很多,因为它不用对instance不为空的情况进行同步。这段代码看起来似乎很完美,但很可惜,还是存在问题。instance=new Singleton() 这并非是一个原子性质的操作,事实上在JVM中做了如下三件事:

 

  1. 给instance分配内存
  2. 调用构造函数初始化成员变量
  3. 将instance对象指向分配的内存(此步骤完成instance即为非空)

但是在 JVM 的即时编译器中存在指令重排序的优化。也就是说上面的第二步和第三步的顺序是不能保证的,最终的执行顺序可能是 1-2-3 也可能是 1-3-2。如果是后者,则在 3 执行完毕、2 未执行之前,被线程二抢占了,这时 instance 已经是非 null 了(但却没有初始化),所以线程二会直接返回 instance,然后使用,然后顺理成章地报错。

 

四、双重检验锁(volatile)

 

public class Singleton {
    private volatile static Singleton instance; //声明成 volatile
    private Singleton (){}

    public static Singleton getSingleton() {
        if (instance == null) {                         
            synchronized (Singleton.class) {
                if (instance == null) {       
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
   
}

 

介绍:使用 volatile 的原因是可以保证线程在本地不会存有 instance 的副本,每次都是去主内存中读取。这里使用volatile的一个主要原因就是volatile可以禁止JVM指令重排序。

 

五、饿汉式

 

public class Singleton{
    //类加载时就初始化
    private static final Singleton instance = new Singleton();
    
    private Singleton(){}

    public static Singleton getInstance(){
        return instance;
    }
}

 

 介绍:这种方法非常简单,因为单例的实例被声明成 static 和 final 变量了,在第一次加载类到内存中时就会初始化,所以创建实例本身是线程安全的。缺点是它不是一种懒加载模式。比如 Singleton 实例的创建是依赖参数或者配置文件的,在 getInstance() 之前必须调用某个方法设置参数给它,那样这种单例写法就无法使用了。

 

五、静态内部类

 

public class Singleton {  
    private static class SingletonHolder {  
        private static final Singleton INSTANCE = new Singleton();  
    }  
    private Singleton (){}  
    public static final Singleton getInstance() {  
        return SingletonHolder.INSTANCE; 
    }  
}

 

 介绍:这种写法仍然使用JVM本身机制保证了线程安全问题;由于 SingletonHolder 是私有的,除了 getInstance() 之外没有办法访问它,因此它是懒汉式的;同时读取实例的时候不会进行同步,没有性能缺陷;也不依赖 JDK 版本。

 

六、枚举

 

public enum EasySingleton {
	
	INSTANCE;
	
	EasySingleton(){
		
	}
	
}

 

介绍:我们可以通过EasySingleton.INSTANCE来访问实例,这比调用getInstance()方法简单多了。创建枚举默认就是线程安全的,所以不需要担心double checked locking,而且还能防止反序列化导致重新创建新的对象。

 

 

<think>好的,我现在需要回答用户关于Java标准单例模式实现方式的问题。首先,用户提到他们参考了站内的两篇引用内容,分别是关于枚举实现的优势以及饿汉式和懒汉式模式的示代码。我需要结合这些信息,同时确保回答符合用户的要求,即结构清晰、分步解释,并且正确引用提供的资料。 首先,我应该回顾单例模式的基本概念,确保用户理解这是用来确保一个类只有一个实,并提供全局访问点。然后,根据引用中的内容,Java中常见的实现方式包括饿汉式、懒汉式、双重检查锁和枚举实现。用户提到的引用[1]强调了枚举实现的优势,而引用[2]则展示了饿汉式和懒汉式的代码示。 接下来,我需要分别解释每种实现方式: 1. 饿汉式:类加载时就初始化实,线程安全,但可能造成资源浪费。引用[2]中的代码示展示了这一点,其中静态变量xh在类加载时就被初始化。 2. 懒汉式:延迟加载,但在多线程环境下需要同步处理,否则可能产生多个实。用户提供的代码中,懒汉式可能没有同步机制,需要指出线程不安全的问题,并给出改进方法,比如加synchronized关键字,或者使用双重检查锁。 3. 双重检查锁:结合懒加载和线程安全,减少同步开销。需要详细说明volatile关键字的作用,以及为什么需要两次检查实是否为null。 4. 枚举实现:引用[1]中提到枚举实现是线程安全且防止反射攻击的最佳方式。需要解释枚举的特性,如实由JVM保证唯一,并且序列化机制天然支持。 在编写过程中,要确保代码示符合用户给出的格式要求,即行内公式用$...$,独立公式用$$...$$,但这里主要是代码块,所以要注意语法正确如,Java代码需要用三个反引号包裹,并注明语言类型。 此外,用户要求生成相关问题,我需要根据回答内容提出三个相关问题,帮助用户进一步学习,比如性能比较、防止反射攻击的方法,以及枚举实现的其他优势。 需要检查是否有遗漏的实现方式,或者是否有其他需要注意的细节,比如序列化问题、反射攻击的防范,这些在枚举实现中已经处理,但在其他实现中可能需要额外代码,比如readResolve方法。 最后,确保回答结构清晰,分点说明,每个实现方式都有代码示和优缺点分析,并且正确引用提供的资料,如引用[1]和引用[2]中的相关内容。同时,语言要使用中文,避免使用Markdown格式,但代码块需要正确标示。</think>Java中标准单例模式的实现主要有以下四种方式,每种方式针对不同场景提供解决方案: ### 一、饿汉式(Eager Initialization) ```java public class Singleton { // 类加载时直接初始化(静态变量) private static final Singleton INSTANCE = new Singleton(); // 私有构造器防止外部实化 private Singleton() {} public static Singleton getInstance() { return INSTANCE; } } ``` * **优点**:线程安全,实现简 * **缺点**:类加载时即创建实,可能造成资源浪费[^2] ### 二、懒汉式(Lazy Initialization) ```java public class Singleton { private static Singleton instance; private Singleton() {} // 同步方法保证线程安全 public static synchronized Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } } ``` * **优点**:延迟初始化 * **缺点**:同步锁影响性能(每次调用都加锁) ### 三、双重检查锁(Double-Checked Locking) ```java public class Singleton { private static volatile Singleton instance; private Singleton() {} public static Singleton getInstance() { if (instance == null) { // 第一次检查 synchronized (Singleton.class) { // 加锁 if (instance == null) { // 第二次检查 instance = new Singleton(); } } } return instance; } } ``` * **优势**:线程安全且性能优化 * **关键点**:`volatile`关键字防止指令重排序[^1] ### 四、枚举实现(推荐方式) ```java public enum Singleton { INSTANCE; public void doSomething() { // 业务方法 } } ``` * **绝对优势**: 1. 天然防止反射攻击 2. 自动处理序列化/反序列化 3. 线程安全由JVM保证[^1] * **应用场景**:Spring框架的Bean管理、数据库连接池 ### 性能对比表 | 实现方式 | 线程安全 | 延迟加载 | 防反射 | 序列化安全 | |------------|------|------|-----|-------| | 饿汉式 | ✔ | ✘ | ✘ | ✘ | | 懒汉式 | ✔ | ✔ | ✘ | ✘ | | 双重检查锁 | ✔ | ✔ | ✘ | ✘ | | **枚举实现** | ✔ | ✘ | ✔ | ✔ |
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值