【设计模式】单例模式:确保类只有一个实例

简介: 【设计模式】单例模式:确保类只有一个实例

软件设计中,单例模式是一种常见的设计模式,它保证一个类只有一个实例,并提供一个全局访问点来访问该实例。单例模式通常用于管理全局状态、资源共享、日志记录等场景。在本文中,我们将深入探讨单例模式的实现方式、使用场景以及一些注意事项。


实现方式

在 Java 中,实现单例模式的常用方式包括:

  1. 饿汉式(Eager Initialization):在类加载时就创建实例,并在静态成员变量中持有该实例。这种方式简单直接,但如果实例不被使用,会造成资源浪费。
public class Singleton {
    private static final Singleton instance = new Singleton();
 
    private Singleton() {}
 
    public static Singleton getInstance() {
        return instance;
    }
}


2.懒汉式(Lazy Initialization):在首次访问时才创建实例。这种方式延迟了实例的创建,但需要考虑线程安全性。

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


3.双重检查锁(Double-Checked Locking):结合了饿汉式和懒汉式的优点,在首次访问时延迟创建实例,并使用双重检查锁机制确保线程安全。

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;
    }
}


4.静态内部类(Static Inner Class):利用类加载机制保证线程安全,且实现简单优雅。当 Singleton 类加载时,静态内部类 SingletonHolder 不会被加载,只有在调用 getInstance() 方法时才会加载 SingletonHolder 类,从而实现懒加载。

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


使用场景

单例模式适用于以下场景:

  • 资源管理:例如数据库连接池、线程池等,通过单例模式可以确保全局只有一个资源管理实例,避免资源浪费和竞争条件。
  • 配置信息:应用程序的全局配置信息可以通过单例模式进行管理,方便访问和修改。
  • 日志记录:单例模式可以用于记录应用程序的日志信息,确保所有日志记录都写入同一个日志文件。
  • 缓存管理:例如对象池、图片缓存等,单例模式可以确保全局只有一个缓存管理实例,避免数据一致性问题。


注意事项

在使用单例模式时需要注意以下几点:

  • 线程安全性:在多线程环境下,需要确保单例实例的创建和访问是线程安全的,可以使用同步机制或者线程安全的初始化方式。
  • 序列化和反序列化:如果单例类需要支持序列化和反序列化,需要实现 Serializable 接口,并且重写 readResolve() 方法,确保反序列化时返回同一个实例。
  • 类加载器:在某些情况下,如果存在多个类加载器,可能会导致单例类被加载多次,从而破坏单例模式。需要注意类加载器的使用和管理。
  • 内存泄漏:如果单例实例长时间持有外部资源或者引用,可能会导致内存泄漏。在不需要使用单例实例时,应该及时释放资源或者引用。


总结

单例模式是一种常见的设计模式,它可以确保一个类只有一个实例,并提供一个全局访问点来访问该实例。在实际应用中,可以根据具体场景选择不同的实现方式,并注意线程安全性、序列化和反序列化、类加载器等问题。合理使用单例模式可以提高代码的可维护性和性能,并且降低资源消耗。

相关实践学习
通过日志服务实现云资源OSS的安全审计
本实验介绍如何通过日志服务实现云资源OSS的安全审计。
相关文章
|
9月前
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
347 15
|
11月前
|
设计模式 安全 Java
设计模式:单例模式
单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。它通过私有化构造函数、自行创建实例和静态方法(如`getInstance()`)实现。适用于数据库连接池、日志管理器等需要全局唯一对象的场景。常见的实现方式包括饿汉式、懒汉式、双重检查锁、静态内部类和枚举。线程安全问题可通过`synchronized`或双重检查锁解决,同时需防止反射和序列化破坏单例。优点是避免资源浪费,缺点是可能增加代码耦合度和测试难度。实际开发中应优先选择枚举或静态内部类,避免滥用单例,并结合依赖注入框架优化使用。
|
10月前
|
设计模式 存储 安全
设计模式-单例模式练习
单例模式是Java设计模式中的重要概念,确保一个类只有一个实例并提供全局访问点。本文详解单例模式的核心思想、实现方式及线程安全问题,包括基础实现(双重检查锁)、懒汉式与饿汉式对比,以及枚举实现的优势。通过代码示例和类图,深入探讨不同场景下的单例应用,如线程安全、防止反射攻击和序列化破坏等,展示枚举实现的简洁与可靠性。
175 0
|
12月前
|
设计模式 存储 安全
设计模式2:单例模式
单例模式是一种创建型模式,确保一个类只有一个实例,并提供全局访问点。分为懒汉式和饿汉式: - **懒汉式**:延迟加载,首次调用时创建实例,线程安全通过双重检查锁(double check locking)实现,使用`volatile`防止指令重排序。 - **饿汉式**:类加载时即创建实例,线程安全但可能浪费内存。 示例代码展示了如何使用Java实现这两种模式。
273 4
|
设计模式 存储 前端开发
前端必须掌握的设计模式——单例模式
单例模式是一种简单的创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。适用于窗口对象、登录弹窗等场景,优点包括易于维护、访问和低消耗,但也有安全隐患、可能形成巨石对象及扩展性差等缺点。文中展示了JavaScript和TypeScript的实现方法。
579 13
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
197 2
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
9月前
|
设计模式 Java 数据库连接
【设计模式】【创建型模式】工厂方法模式(Factory Methods)
一、入门 什么是工厂方法模式? 工厂方法模式(Factory Method Pattern)是一种创建型设计模式,它定义了一个用于创建对象的接口,但由子类决定实例化哪个类。工厂方法模式使类的实例化延迟
285 16
|
9月前
|
设计模式 负载均衡 监控
并发设计模式实战系列(2):领导者/追随者模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第二章领导者/追随者(Leader/Followers)模式,废话不多说直接开始~
275 0
|
9月前
|
设计模式 监控 Java
并发设计模式实战系列(1):半同步/半异步模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第一章半同步/半异步(Half-Sync/Half-Async)模式,废话不多说直接开始~
298 0