设计模式 -创建型模式之单例模式的五种实现

简介: 设计模式 -创建型模式之单例模式的五种实现

单例模式(Singleton)


单例模式是在 GOF的23种设计模式里较为简单的一种,下面引用百度百科介绍:


单例模式,是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中,应用该模式的类一个类只有一个实例。即一个类只有一个对象实例


许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为。比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息。这种方式简化了在复杂环境下的配置管理。


在Java中,确保一个类只有一个对象实例可以通过权限的修饰来实现。


单例模式 - 饿汉模式


单例模式的饿汉模式指全局的单例实例在第一次被使用时构建。


具体实现:


// 单例模式的饿汉模式实现
  public class Singleton {
    private final static Singleton SINGLETON= new Singleton();
    // Private constructor suppresses   
    private Singleton() {}
    // default public constructor
    public static Singleton getInstance() {
        return SINGLETON;
    }
  }


在饿汉模式实现方式中,程序的主要特点是:


  1. 私有构造方法
  2. 私有静态属性,维护自身实例
  3. 静态服务方法,获取实例
  4. 初始化时候创建,消耗初始化系统资源

单例模式 - 懒汉模式 - 普通


懒汉模式,也是最常用的形式,饿汉模式让程序在初始化时候进行加载,有时为了节约资源,我们需要在需要的时候进行加载,这时候我们可以使用懒汉模式。


具体实现:


public class SingletonLayload {    
    // 私有化自身类对象
    private static SingletonLayload SINGLETON;
    // 私有化构造方法
    private SingletonLayload() {}
    // 静态方法获取实例
    public static SingletonLayload getInstance() {
        if(SINGLETON== null ) {
            SINGLETON= new SingletonLayload();
        }
        return SINGLETON;
    }
}


单例模式 - 懒汉模式 - 同步锁


在多线程的环境中,简单的单例模式将会出现问题,试想在上面的懒汉模式中,如果多线程并发执行getInstance(),当线程A执行到:


INSTANCE = new SingletonLayload();


却还没有执行完毕时,线程B执行到if(INSTANCE == null ),此时就无法保证单例

特性。


因此在多线程环境中,单例模式需要使用同步锁确保实现真正的单例。


具体实现:


public class SingletonLayloadSyn {
    // 私有化自身类对象
    private static SingletonLayloadSyn SINGLETON;
    // 私有化构造方法
    private SingletonLayloadSyn() {}
    // 静态方法获取实例
    public static synchronized SingletonLayloadSyn getInstance() {
        if(SINGLETON == null ) {
            SINGLETON = new SingletonLayloadSyn();
        }
        return SINGLETON;
    }
}


通过在getInstance()方法上添加 synchronized 关键字可以解决多线程带来的问题。



单例模式 - 懒汉模式 - 双重校验锁


使用上面的( 多线程下 - 懒汉模式 - 同步锁)方式在解决多线程问题时虽然可以达到确保线程安全的目的,但是使用了synchronized关键字之后在需要多次调用时,会让代码的执行效率大大降低。那么有没有在确保线程安全的同时又可以兼顾效率的方法呢?


具体实现:


public class SingletonLayLoadSynDCL {
    // 私有化自身类对象
    private static SingletonLayLoadSynDCL SINGLETON;
    // 私有化构造方法
    private SingletonLayLoadSynDCL() {
    }
    public static  SingletonLayLoadSynDCL getInstance() {
        if (SINGLETON == null) {
            synchronized(SingletonLayLoadSynDCL.class) {
                SINGLETON = new SingletonLayLoadSynDCL();
            }
        }
        return SINGLETON;
    }
}


使用 synchronized 确保线程安全,在SINGLETON 为 null 时才进行创建实例,但是仍然不能 保证在实例未创建完成时候有新的线程执行到 if (SINGLETON == null);因此,仍然不够安全。

修改 getInstance()方法。


具体实现:


public class SingletonLayLoadSynDCL {
    // 私有化自身类对象
    private static SingletonLayLoadSynDCL SINGLETON;
    // 私有化构造方法
    private SingletonLayLoadSynDCL() {
    }
    // 使用双重校验锁确保线程安全的同时兼顾执行效率
    public static SingletonLayLoadSynDCL getInstance() {
        if (SINGLETON == null) { // 第一重检查
            synchronized (SingletonLayLoadSynDCL.class) {
                if (SINGLETON == null) { //第二重检查
                    SINGLETON = new SingletonLayLoadSynDCL();
                }
            }
        }
        return SINGLETON;
    }
}


看似完美的双检查模式,在理论上是没有问题的。但是在实际的情况里,有可能发生在没有构造完毕的情况下SINGLETON 引用已经不是 NULL 的情况,这时候如果有其他线程执行到if (SINGLETON == null) { // 第一重检查则会获取到一个不正确的 SINGLETON 引用。这是由于JVM 的无序写入引起的。


幸好,在 JDK1.5 之后,提供了volatile关键字,用于确保被修饰的变量的读写不允许被控制。因此修改上面具体实现为:


/**
 * <p>
 * 使用双重校验锁以及volatile关键字确保线程安全的同时兼顾执行效率
 * @author  niujinpeng
 */
public class SingletonLayLoadSynDCL {
    // 私有化自身类对象
    //  private static SingletonLayLoadSynDCL SINGLETON;
    private volatile static SingletonLayLoadSynDCL SINGLETON;
    // 私有化构造方法
    private SingletonLayLoadSynDCL() {}
    // 使用双重校验锁确保线程安全的同时兼顾执行效率
    public static SingletonLayLoadSynDCL getInstance() {
        if (SINGLETON == null) {
            synchronized (SingletonLayLoadSynDCL.class) {
                if (SINGLETON == null) {
                    SINGLETON = new SingletonLayLoadSynDCL();
                }
            }
        }
        return SINGLETON;
    }
}


单例模式 - 懒汉模式 - 内部类


除了使用上面的懒汉模式实现方式之外,在解决多线程问题中,《Effective Java》的作者给出了另外一种保证线程安全且兼顾效率的方式,利用了静态内部类以及类加载特性实现。静态内部类只有在调用时才会加载,而静态属性随着类的加载而加载,类的加载初始化只会有一次。因此保证了获取实例的唯一性。


具体实现:


package cn.snowflow.pattern.singleton;
/**
 * <p>
 * 利用静态内部类实现线程安全且兼顾效率的单例模式
 * @author  niujinpeng
 */
public class SingletonLayloadSynSafe {
    //静态内部类
    public static class SingletonHolder{
        static final SingletonLayloadSynSafe INSTANCE = 
            new SingletonLayloadSynSafe();
    }
    // 私有化构造方法
    private SingletonLayloadSynSafe() {}
    // 公有方法获取实例
    public static SingletonLayloadSynSafe getInstance() {
        return SingletonHolder.INSTANCE;
    }
}


如果使用单例模式-饿汉模式,推荐【单例模式 - 饿汉模式】。


如果使用单例模式-懒汉模式,推荐【单例模式 - 懒汉模式 - 内部类 】。

相关文章
|
5月前
|
设计模式 Java 数据库连接
【设计模式】【创建型模式】工厂方法模式(Factory Methods)
一、入门 什么是工厂方法模式? 工厂方法模式(Factory Method Pattern)是一种创建型设计模式,它定义了一个用于创建对象的接口,但由子类决定实例化哪个类。工厂方法模式使类的实例化延迟
143 16
|
5月前
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
171 15
|
5月前
|
设计模式 JavaScript Java
【设计模式】【创建型模式】原型模式(Prototype)
一、入门 什么是原型模式? 原型模式(Prototype Pattern)是一种创建型设计模式,它通过复制现有对象来创建新对象,而不是通过实例化类。 原型模式的核心是克隆(Clone),即通过复制现有
176 15
|
5月前
|
设计模式 Java Apache
【设计模式】【创建型模式】建造者模式(Builder)
一、入门 什么是建造者模式? 建造者模式(Builder Pattern)是一种创建型设计模式,用于逐步构建复杂对象。 它通过将对象的构建过程与表示分离,使得相同的构建过程可以创建不同的表示。 为什么
195 14
|
5月前
|
设计模式 Java 关系型数据库
【设计模式】【创建型模式】抽象工厂模式(Abstract Factory)
一、入门 什么是抽象工厂模式? 抽象工厂模式是一种创建型设计模式,它提供了一个接口,用于创建相关或依赖对象的家族,而不需要指定具体的类。 简单来说,抽象工厂模式是工厂方法模式的升级版,它能够创建一组相
174 14
|
7月前
|
设计模式 XML Java
设计模式觉醒系列(03)创建型模式的5个设计模式 | 一口气讲全讲透
本文详细介绍了设计模式中的创建型模式,包括建造者模式、原型模式、单例模式、工厂方法模式和抽象工厂模式。创建型模式关注对象的创建过程,隐藏了创建细节,以提高代码的可维护性和可扩展性。通过具体的实战demo和应用场景分析,展示了每种模式的特点和优势。例如,建造者模式适用于复杂对象的分步骤构建;原型模式通过复制对象实现高效复用;单例模式确保全局唯一实例;工厂方法模式和抽象工厂模式则提供了灵活的对象创建机制,支持多类型产品族的生产。这些模式在实际开发中能够简化客户端代码,提升系统灵活性和复用性。
|
6月前
|
设计模式 存储 安全
设计模式-单例模式练习
单例模式是Java设计模式中的重要概念,确保一个类只有一个实例并提供全局访问点。本文详解单例模式的核心思想、实现方式及线程安全问题,包括基础实现(双重检查锁)、懒汉式与饿汉式对比,以及枚举实现的优势。通过代码示例和类图,深入探讨不同场景下的单例应用,如线程安全、防止反射攻击和序列化破坏等,展示枚举实现的简洁与可靠性。
113 0
|
7月前
|
设计模式 安全 Java
设计模式:单例模式
单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。它通过私有化构造函数、自行创建实例和静态方法(如`getInstance()`)实现。适用于数据库连接池、日志管理器等需要全局唯一对象的场景。常见的实现方式包括饿汉式、懒汉式、双重检查锁、静态内部类和枚举。线程安全问题可通过`synchronized`或双重检查锁解决,同时需防止反射和序列化破坏单例。优点是避免资源浪费,缺点是可能增加代码耦合度和测试难度。实际开发中应优先选择枚举或静态内部类,避免滥用单例,并结合依赖注入框架优化使用。
|
8月前
|
设计模式 存储 安全
设计模式2:单例模式
单例模式是一种创建型模式,确保一个类只有一个实例,并提供全局访问点。分为懒汉式和饿汉式: - **懒汉式**:延迟加载,首次调用时创建实例,线程安全通过双重检查锁(double check locking)实现,使用`volatile`防止指令重排序。 - **饿汉式**:类加载时即创建实例,线程安全但可能浪费内存。 示例代码展示了如何使用Java实现这两种模式。
143 4
|
10月前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是"将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。创建型模式分为5种:单例模式、工厂方法模式抽象工厂式、原型模式、建造者模式。
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析

热门文章

最新文章