Java单例模式的写法及优缺点

本文详细介绍了Java中单例模式的五种实现方式:饿汉式、懒汉式(线程不安全)、懒汉式(线程安全)、双检锁/双重校验锁(DCL)以及静态内部类单例模式。每种方式的优缺点进行了分析,特别强调了DCL的线程安全问题及解决方案。最后推荐使用静态内部类单例模式,因其线程安全且资源利用率高。同时,文章提及了防止单例在反序列化时被重新创建的解决方法,即使用readResolve()方法。

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

总结下Java单例模式的几种写法:

1.饿汉式

public class Singleton
{

    private static Singleton instance = new Singleton();

    private Singleton() {}

    public static Singleton getInstance() {
        return instance;
    }

}

优点:实现简单,不存在多线程问题,直接声明一个私有对象,然后对外提供一个获取对象的方法。

缺点:class 类在被加载的时候创建Singleton实例,如果对象创建后一直没有使用,则会浪费很大的内存空间,此方法不适合创建大对象。

2.懒汉式(线程不安全)

public class Singleton
{

    private static Singleton instance = null;

    private Singleton() {}

    public static Singleton getInstance() {

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

}

优点:节省内存空间,在使用的时候才会创建;

缺点:线程不安全,在多线程下,可能会创建多个实例(一定要重视这个问题,有时候如果在单例对象的构造方法中做了某些重要操作,创建多个实例可能会造成可怕后果,如:打开Android的Sqlite数据库连接)。

3.懒汉式(线程安全)

public class Singleton
{

    private static Singleton instance = null;

    private Singleton() {}

    public synchronized static Singleton getInstance() {

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

}

优点:支持多线程,且以懒汉式的方式加载,不浪费内存空间。

缺点:将 synchronized 块加在方法上,会影响并发量,每次调用getInstance()方法都会线程同步,效率十分低下。最重要的是,当创建好实例对象之后,就不必继续进行同步了。

4.懒汉式(double check)(线程安全,推荐)

public class Singleton
{

    private static volatile Singleton instance = null;

    private Singleton() {}

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

}

优点:支持多线程,并发量高,且以懒汉式加载,不浪费内存空间。

缺点:一时找不出缺点,非要说缺点的话,就是实现比较麻烦。

在这里使用volatile会或多或少的影响性能,但考虑到程序的正确性,牺牲这点性能还是值得的。 DCL优点是资源利用率高,第一次执行getInstance时单例对象才被实例化,效率高。缺点是第一次加载时反应稍慢一些,在高并发环境下也有一定的缺陷,虽然发生的概率很小。DCL虽然在一定程度解决了资源的消耗和多余的同步,线程安全等问题,但是他还是在某些情况会出现失效的问题,**也就是DCL失效,在《Java并发编程实践》一书建议用静态内部类单例模式来替代DCL。**其实就是指令重排序的问题,所以使用volatile可以防止指令重排序

DCL失效的原因
LazyLoad,这种技巧很常用,就是指一个类包含某个成员变量,在类初始化的时候并不立即为该变量初始化一个实例,而是等到真正要使用到该变量的时候才初始化之。
例如下面的代码:

class Foo {
  private Resource res = null;
  public Resource getResource() {
    if (res == null)
      res = new Resource();
    return res;
  }
}

在单线程环境下,一切都相安无事,但如果把上面的代码放到多线程环境下运行,那么就可能会出现问题。假设有2条线程,同时执行到了if(res == null),那么很有可能res被初始化2次,为了避免这样的Race Condition,得用synchronized关键字把上面的方法同步起来。代码如下:

Class Foo {
  private Resource res = null;
  public synchronized Resource getResource() {
    if (res == null)
      res = new Resource();
    return res;
  }
}

synchronized过的方法在速度上要比未同步的方法慢上100倍,同时你也发现,只有第一次调用该方法的时候才需要同步,而一旦res初始化完成,同步完全没必要。所以你很快就把代码重构成了下面的样子:

Class Foo {
private Resource res = null;
private Date d = new Data();
  public Resource getResource() {
    if (res == null){                    //(1)
      synchronized(Foo.class){
        if(res == null){
          res = new Resource();         //(2)
		}
	  }
    }
    return res;
  }
}

Double-Checked Locking看起来是非常完美的。但是很遗憾,根据Java的语言规范,上面的代码是不可靠的。

出现上述问题, 最重要的2个原因如下:
1, 编译器优化了程序指令, 以加快cpu处理速度.
2, 多核cpu动态调整指令顺序, 以加快并行运算能力.
问题出现的顺序:
1, 线程A, 发现对象未实例化, 准备开始实例化
2, 由于编译器优化了程序指令, 允许对象在构造函数未调用完前, 将共享变量的引用指向部分构造的对象, 虽然对象未完全实例化, 但已经不为null了.
3, 线程B, 发现部分构造的对象已不是null, 则直接返回了该对象.

一个线程A运行到"这里"时,A的工作区中,肯定已经产生一个Foo对象,而且这时这个对象已经完成了Data d.现在线程A调用时间到,执行权被切换到另一个线程B来执行,会有什么问题呢?如果res不为null,线程B获得了一个res,但可能res.getD()却还没有初始化.
对于"这里"这条语句,线程A还没有离开同步块.因为没有"离开同步块"这个条件,线程A的工作区没有强制与主存储器同步,这时工作区中有两个字段res,d。虽然在线程A的工作区res和d都是完整的,但有JSL没有强制不允许先把res映射到主存储区,如果哪个jvm实现按它的优化方案先把工作存储器中的res同步到主存储器了,这时正好线程B获取了,而d却没有同步过去,那么线程B就获取了res的引用却找不能res.getD()。

5.静态内部类单例模式

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

第一次加载Singleton类时并不会初始化单例,只有第一次调用getInstance方法时虚拟机加载SingletonHolder 并初始化单例 ,这样不仅能确保线程安全也能保证Singleton类实例的唯一性,所以推荐使用静态内部类单例模式。

总结

上述讲的其他几种单例模式实现中,有一种情况下他们会重新创建对象,那就是反序列化,将一个单例实例对象写到磁盘再读回来,从而获得了一个实例。反序列化操作提供了readResolve方法,这个方法可以让开发人员控制对象的反序列化。在上述的几个方法示例中如果要杜绝单例对象被反序列化是重新生成对象,就必须加入如下方法:

private Object readResolve() throws ObjectStreamException{
    return singleton;
}

推荐使用静态内部类实现,如果本身没有序列化的要求,按照之前实例代码的写法即可,如果需要支持序列化,则需要添加readResolve方法,如下

public class Singleton implements Serializable { 
    private Singleton(){
    }
      public static Singleton getInstance(){  
        return SingletonHolder.instance;  
    }  
    private static class SingletonHolder {  
        private static final Singleton instance = new Singleton();  
    }  
 
    private Object readResolve() throws ObjectStreamException {
        return SingletonHolder.instance;
    }
}
<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管理、数据库连接池 ### 性能对比表 | 实现方式 | 线程安全 | 延迟加载 | 防反射 | 序列化安全 | |------------|------|------|-----|-------| | 饿汉式 | ✔ | ✘ | ✘ | ✘ | | 懒汉式 | ✔ | ✔ | ✘ | ✘ | | 双重检查锁 | ✔ | ✔ | ✘ | ✘ | | **枚举实现** | ✔ | ✘ | ✔ | ✔ |
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

进击的代码家

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值