Java设计模式详解:单例模式

news/2024/5/4 7:35:49

设计模式详解:单例模式


文章目录

  • 设计模式详解:单例模式
  • 一、单例模式的原理
  • 二、单例模式的实现推荐
    • 1、饿汉模式
    • 2、静态内部类
  • 三、单例模式的案例
  • 四、单例模式的使用场景推荐
  • 总结


一、单例模式的原理

单例模式听起来很高大上,但其实它的核心思想很简单,就是确保一个类只有一个实例,并提供一个全局访问点。单例模式的原理其实不难理解。想象一下,你有一个类,这个类负责某种特定的资源或功能,而这个资源或功能在整个应用程序中只需要一个实例就足够了。这时,你就可以使用单例模式来确保这个类只有一个实例,并且这个实例可以被整个应用程序访问。

要实现单例模式,通常需要做到以下几点:

  1. 私有化构造函数,防止外部通过new关键字创建实例。
  2. 提供一个静态的私有变量来保存类的唯一实例。
  3. 提供一个公共的静态方法来获取类的唯一实例,如果实例不存在则创建它。

二、单例模式的实现推荐

1、饿汉模式

饿汉式在类加载时就完成了初始化,所以类加载较慢,但获取对象的速度快。可能会造成一些资源的损失浪费,但是不需要开发者关心线程安全之类的问题。它有以下优点:

  1. 线程安全:由于实例的初始化在类加载时就完成了,因此它是线程安全的,多个线程同时调用getInstance()方法时,不会存在竞态条件。
  2. 简单明了:实现方式简单直观,易于理解。

代码示例

public class Singleton {  private Singleton() {  // 私有化构造函数  }  // 静态内部类  private static class SingletonHolder {  private static final Singleton INSTANCE = new Singleton();  }  public static Singleton getInstance() {  return SingletonHolder.INSTANCE;  }  
}

2、静态内部类

静态内部类利用JVM的类加载机制来保证初始化instance时只有一个线程,所以它是线程安全的,同时该实现方式也是延迟加载的,即类加载时不初始化,只有在第一次调用getInstance()方法时才初始化。它有以下优点:

  1. 线程安全:与饿汉式类似,由于使用了静态内部类,在类加载时并不会初始化INSTANCE,而是在第一次调用getInstance()方法时,由JVM来确保线程安全地初始化INSTANCE。
  2. 延迟加载:实现了懒汉式的延迟加载效果,只有在第一次真正需要使用单例时才进行初始化,节省了系统资源。
  3. 双重校验锁效果的简化实现:避免了复杂的双重校验锁实现,降低了出错的可能性,同时保持了线程安全。

静态内部类的实现结合了饿汉式的线程安全和懒汉式的延迟加载的优点,既保证了线程安全,又实现了延迟加载,是一种较为推荐的单例实现方式。

三、单例模式的案例

为了更好地理解单例模式,我们来看一个简单的案例。假设我们有一个配置管理类ConfigManager,这个类负责读取和提供应用程序的配置信息。由于配置信息在应用程序中是全局共享的,所以我们希望ConfigManager在整个应用程序中只有一个实例。

下面是一个简单的单例模式实现:

public class ConfigManager {  // 静态的私有变量保存类的唯一实例  private static ConfigManager instance;  // 私有化构造函数  private ConfigManager() {  // 初始化代码  }  // 公共的静态方法获取类的唯一实例  public static ConfigManager getInstance() {  if (instance == null) {  instance = new ConfigManager();  }  return instance;  }  // 提供读取配置信息的方法  public String getConfigValue(String key) {  // 实现读取配置的逻辑  return "value from config";  }  
}

在上面的代码中,我们通过将构造函数私有化来防止外部直接创建ConfigManager的实例。然后,我们提供了一个公共的静态方法getInstance()来获取类的唯一实例。当第一次调用这个方法时,它会创建ConfigManager的实例并保存在instance变量中;之后再次调用时,就直接返回已经创建的实例。

四、单例模式的使用场景推荐

  1. 日志记录器:单例模式能够确保整个应用程序使用同一个日志记录器实例。这有助于统一管理和配置日志记录器的行为,包括日志级别、输出格式、存储位置等,使得日志信息能够按照预定的规范进行记录,方便后续的查看和分析。
  2. 配置管理类:在应用程序中,通常会有一些全局的配置信息,如数据库连接信息、系统参数等。这些配置信息在整个应用程序中都需要访问,且应该保持一致。通过使用单例模式,可以确保配置管理类只有一个实例,从而避免配置信息的重复加载和不一致问题。
  3. 数据库连接池:数据库连接是应用程序中经常使用的资源,频繁地创建和销毁数据库连接会消耗大量的系统资源,并可能导致性能问题。通过使用单例模式实现数据库连接池,可以确保整个应用程序使用一个共享的数据库连接池实例。这样,连接池可以管理一定数量的数据库连接,并根据需要进行复用,从而提高系统性能和响应速度。
  4. 线程池:线程是应用程序中执行任务的基本单位。频繁地创建和销毁线程会浪费系统资源,并可能导致线程管理混乱。通过使用单例模式实现线程池,可以管理和复用线程资源,减少线程的创建和销毁开销。线程池可以维护一定数量的线程,并根据任务需求进行调度和执行,从而提高系统的并发性能和响应速度。
  5. SpringBoot使用单例模式:在Spring Boot框架中,单例模式得到了广泛的应用。通过使用@Component、@Service、@Controller等注解,可以将Bean托管到Spring的容器中进行管理。这些注解标识的类在Spring容器中默认是单例的,即整个Spring IoC容器中只会存在一个实例。这种管理方式简化了对象的创建和管理,使得开发者可以更加专注于业务逻辑的实现,而无需关心对象的生命周期和依赖关系。

单例模式的使用场景非常广泛,除了上述列的之外,还有如缓存管理、消息队列、定时任务管理等。然而,需要注意的是,虽然单例模式在某些场景下非常有用,但它并不适用于所有情况。在设计系统时,应根据具体需求和场景来选择合适的设计模式,以确保系统的可维护性、可扩展性和性能。


总结

通过掌握单例模式的原理、案例和使用场景,我们可以更加灵活地运用它来解决实际开发中的问题。希望这篇文章能够帮助大家更好地理解和应用单例模式,为后续的Java学习和实践打下坚实的基础。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.cpky.cn/p/11354.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈,一经查实,立即删除!

相关文章

Godot 4 教程《勇者传说》依赖注入 学习笔记(0):环境配置

文章目录 前言相关地址环境配置初始化环境配置文件夹结构代码结构代码运行 资源文件导入像素风格窗口环境设置背景设置,Tileap使用自动TileMap 人物场景动画节点添加站立节点添加移动动画添加 通过依赖注入获取Godot的全局属性项目声明 当前项目逻辑讲解角色下降添加代码位置问…

外包干了5天,技术退步明显.......

先说一下自己的情况,大专生,18年通过校招进入杭州某软件公司,干了接近4年的功能测试,今年年初,感觉自己不能够在这样下去了,长时间呆在一个舒适的环境会让一个人堕落! 而我已经在一个企业干了四年的功能测…

git-怎样把连续的多个commit合并成一个?

Git怎样把连续的多个commit合并成一个? Git怎样把连续的多个commit合并成一个? 参考URL: https://www.jianshu.com/p/5b4054b5b29e 查看git日志 git log --graph比如下图的commit 历史,想要把bai “Second change” 和 “Third change” 这…

每日一题--- 环形链表[力扣][Go]

环形链表 题目:142. 环形链表 II 给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。 如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给…

通过node 后端实现颜色窃贼 (取出某个图片的主体rgb颜色 )

1.需求 我前端轮播图的背景色 想通过每一张轮播图片的颜色作为背景色 这样的话 需要通过一张图片 取出图片的颜色 这个工作通过前端去处理 也可以通过后端去处理 前端我试了试 color-thief 的插件 但是 这个插件是基于canvas 的模式来的 我需要在小程序中使用这个插件 而且是…

qt事件机制学习笔记

实现闹钟功能 #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget), speecher(new QTextToSpeech(this)) //给语音播报者实例化空间 {ui->setupUi(this); }Widget::~Widget() {delete …