软件设计模式系列之六——单例模式

2023-09-15 12:44:12

1 模式的定义

单例模式(Singleton Pattern)是一种常见的创建型设计模式,其主要目的是确保一个类只有一个实例,并提供一个全局访问点来获取该实例。这意味着无论何时何地,只要需要该类的实例,都会返回同一个实例,而不是创建多个相同的实例。单例模式通常用于管理全局状态、资源共享或限制某些资源的访问。

2 举例说明

在日常生活中,随处可见单例模式的例子,比如你家中有一台电视,通常只需要一个遥控器来控制它。无论家里谁想看电视,都会使用同一个遥控器,而且遥控器只能让家里人轮流使用,也就是不能有多个人同时使用遥控器控制电视。这个遥控器就是一个单例,因为它确保只有一个实例存在,并且提供了一个全局的访问点,以便你可以随时使用它。

3 结构

单例模式的结构包括以下要素:
在这里插入图片描述

  • 单例类(Singleton Class):单例模式的核心是单例类,它负责管理唯一的实例。通常,这个类会将其构造函数设为私有,以防止外部直接实例化多个对象。单例类会定义一个静态方法或变量来获取或创建唯一的实例。

  • 私有构造函数(Private Constructor):单例类的构造函数通常会被设置为私有,这样外部无法直接实例化这个类。私有构造函数的目的是确保只有单例类内部可以创建类的实例。

  • 静态成员变量(Static Member Variable):单例类会包含一个私有的静态成员变量,用于保存唯一的实例。这个成员变量通常被命名为 instance 或类似的名称。

  • 静态方法(Static Method):单例类会提供一个公共的静态方法,通常命名为 getInstance() 或类似的名称,用于获取或创建唯一的实例。这个方法会检查是否已经存在实例,如果存在则返回现有实例,否则创建一个新的实例并返回它。

单例模式的关键是将构造函数私有化,以确保只有一个实例,并提供一个全局的方法来获取这个实例,以实现全局唯一性。这种结构确保了在应用程序中只有一个实例存在,无论何时何地都可以访问这个实例,从而实现了单例模式的设计目标。

4 实现步骤

实现单例模式的关键步骤通常包括以下几个:

  1. 将构造函数私有化(Private Constructor):在单例模式中,首先需要将单例类的构造函数设为私有,以防止外部直接实例化多个对象。这是确保只有一个实例的重要步骤。

  2. 创建一个私有的静态成员变量(Private Static Member Variable):单例类内部通常会包含一个私有的静态成员变量,用于保存唯一的实例。这个变量通常被命名为 instance 或类似的名称。

  3. 提供一个公共的静态方法(Public Static Method):单例类会提供一个公共的静态方法,通常命名为 getInstance() 或类似的名称,用于获取或创建唯一的实例。这个方法会检查是否已经存在实例,如果存在则返回现有实例,否则创建一个新的实例并返回它。

  4. 在获取实例时进行实例化(Lazy Initialization):在 getInstance() 方法中,需要检查 instance 是否为 None,如果为 None,则创建一个新的实例并将其赋值给 instance,否则直接返回 instance。这确保了实例在需要时才会被创建,避免了不必要的开销。

  5. 处理多线程环境(Thread Safety):如果应用程序可能在多线程环境下使用单例类,需要考虑线程安全性。可以使用加锁机制来确保在多线程环境下也只有一个实例被创建。

5 代码实现

在Java中,可以使用懒汉式和饿汉式两种方式来实现单例模式。下面分别给出这两种方式的示例代码:
在这里插入图片描述

懒汉式单例模式
在懒汉式中,实例是在首次被请求时才创建。

public class LazySingleton {
    private static LazySingleton instance;

    private LazySingleton() {
        // 私有构造函数,防止外部实例化
    }

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

在懒汉式中,getInstance 方法首先检查实例是否已经创建。如果没有创建实例,则创建一个新的实例并返回。这种实现延迟了实例的创建,只有在需要时才会创建。

饿汉式单例模式
在饿汉式中,实例在类加载时就被创建,无论是否需要。

public class EagerSingleton {
    private static final EagerSingleton instance = new EagerSingleton();

    private EagerSingleton() {
        // 私有构造函数,防止外部实例化
    }

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

在饿汉式中,实例在类加载时就被创建,因此无论何时需要实例,都可以立即返回。这种实现简单且线程安全,但可能会造成资源浪费,因为实例会在应用程序启动时就被创建。

需要注意的是,懒汉式在多线程环境下需要额外的同步措施来确保线程安全,而饿汉式天生是线程安全的。选择使用哪种方式取决于具体的需求和性能考虑。

6 典型应用场景

单例模式在各种应用场景中都有广泛的应用,主要用于确保一个类只有一个实例,并提供全局访问点。以下是一些常见的单例模式应用场景:
在这里插入图片描述

数据库连接池:在大多数应用程序中,与数据库的交互是常见的操作。为了提高性能和资源利用率,应用程序通常会使用数据库连接池来管理数据库连接。单例模式可以用于确保只有一个数据库连接池的实例存在,以避免多次创建和销毁数据库连接。

线程池:线程池用于管理和控制线程的执行。通过使用单例模式,可以确保只有一个线程池实例,从而更有效地管理并发执行的任务。

配置管理:在应用程序中,通常需要读取和管理配置信息,例如数据库连接参数、应用程序设置等。单例模式可用于存储和管理这些配置数据,以确保在整个应用程序中使用相同的配置。

日志记录器:在应用程序中记录日志是一项重要的任务,通常会使用日志记录器来处理日志信息。通过单例模式,可以确保只有一个日志记录器实例,以避免多次初始化和配置日志记录器。

窗口管理器:在图形用户界面应用程序中,窗口管理器用于管理应用程序窗口的创建、销毁和切换。单例模式可用于确保只有一个窗口管理器实例,以维护窗口状态和顺序。

单例模式在需要确保全局唯一性、资源共享、全局访问和状态管理的各种应用场景中非常有用。它可以帮助简化代码、提高性能,并确保应用程序的一致性。然而,需要谨慎使用,以避免引入全局状态和多线程问题。

7 优缺点

优点:
全局访问点:通过单例模式,可以在应用程序的任何地方轻松访问相同的实例。
资源共享:单例模式可用于管理共享的资源,例如数据库连接、线程池等,以提高性能和资源利用率。
避免重复创建:单例模式确保只有一个实例,避免了重复创建对象的开销。
缺点:
可能引入全局状态:过度使用单例模式可能导致全局状态,使得代码难以维护和测试。
不适用于多线程环境:如果不正确地实现单例模式,可能会导致多线程竞态条件,需要额外的同步机制来解决。

8 类似模式

在软件开发中,单例模式和原型模式通常在创建和管理"bean"(也称为对象或组件)时发挥重要作用,但它们在此上下文中有不同的用途和应用场景。

单例模式在bean的创建中的应用:

Spring框架中的单例bean:在Spring框架中,默认情况下,Spring容器会将Bean配置为单例(Singleton)。这意味着每个bean在应用程序中只有一个实例,并且Spring容器负责管理这些单例bean的生命周期。这种单例模式的应用确保了全局唯一性,并且可以节省资源和提高性能。
原型模式在bean的创建中的应用:

原型范围的Spring bean:在Spring框架中,你可以将bean配置为原型(Prototype)范围,这意味着每次从Spring容器请求该bean时,都会创建一个新的实例。原型模式的应用适用于那些需要频繁创建新实例的场景,例如HTTP请求的处理,每个请求需要一个新的bean实例以避免状态共享。
关系和应用场景:

单例模式通常用于那些需要确保全局唯一性的bean,例如服务层的单例组件、数据库连接池、配置管理器等。它适用于那些需要共享状态或资源的情况。

原型模式通常用于那些需要频繁创建新实例的bean,例如Web应用程序中的请求处理器、线程池中的任务、HTTP会话管理器等。它适用于那些需要隔离状态或资源的情况。

在Spring框架中,你可以根据bean的具体需求将它们配置为单例或原型范围,以满足应用程序的不同要求。这两种模式有各自的优势和适用场景,可以根据业务逻辑和性能要求来选择合适的范围。

9 小结

单例模式是一种有用的设计模式,用于确保一个类只有一个实例,并提供全局访问点。它在多种应用场景中都有用武之地,但需要小心使用,以避免引入全局状态和多线程问题。通过将构造函数私有化、使用静态变量保存实例以及提供一个静态方法来获取实例,可以实现单例模式。在设计应用程序时,要考虑是否需要使用单例模式来满足特定的需求。

更多推荐

C++关键词探索:理解变量、函数参数、函数返回值以及类成员函数的修饰符

在C++编程中,我们经常会遇到一些关键词,它们可以用来修饰变量、函数参数、函数返回值以及类的成员函数。这些关键词包括const、static、volatile、mutable、signed、unsigned、long、short、virtual、explicit、inline和friend。让我们一起来深入理解一下这些

基于SSM的高校教学业绩信息管理系统设计与实现

末尾获取源码开发语言:JavaJava开发工具:JDK1.8后端框架:SSM前端:采用JSP技术开发数据库:MySQL5.7和Navicat管理工具结合服务器:Tomcat8.5开发软件:IDEA/Eclipse是否Maven项目:是目录一、项目简介二、系统功能三、系统项目截图​编辑四、核心代码登录相关文件上传封装五、

Vue路由及Node.js环境搭建

1.介绍什么是Vue.js和Node.js?Vue.js和Node.js是两个不同的技术,分别用于前端和后端开发,具有不同的用途和功能:Vue.js:Vue.js是一款流行的前端JavaScript框架,也被称为渐进式框架。它由尤雨溪开发,并由社区支持和维护。Vue.js主要用于构建现代、交互式的Web用户界面。它的核

React中组件通信02——消息订阅与发布、取消订阅以及卸载组件时取消订阅

React中组件通信02——消息订阅与发布、取消订阅以及卸载组件时取消订阅1.前言1.1使用props通信1.2关于useEffect2.安装pubsub-js3.消息订阅与发布3.1简单例子-13.2简单例子-2(完善、优化)——订阅消息+使用消息4.取消订阅4.1取消单个topic4.2取消多个或更多语法4.3卸载

LeetCode 面试题 04.09. 二叉搜索树序列

文章目录一、题目二、C#题解一、题目从左向右遍历一个数组,通过不断将其中的元素插入树中可以逐步地生成一棵二叉搜索树。给定一个由不同节点组成的二叉搜索树root,输出所有可能生成此树的数组。点击此处跳转题目。示例1:输入:root=[2,1,3]输出:[[2,1,3],[2,3,1]]解释:数组[2,1,3]、[2,3,

macOS 12 Monterey:一次全新的跨设备协作体验

macOS12Monterey是苹果公司的一次重大突破,它打破了设备间的壁垒,将不同设备无缝地连接在一起,极大地提升了用户的工作效率和娱乐体验。Monterey带来了通用控制、AirPlay、捷径等新功能,以及一些实用的新小功能。安装:macOS12Montereyv12.6.9正式版功能特点通用控制macOS12Mo

会“穿墙术”的神奇材料 ——超固体

超固体(supersolid)是一种具备超流特性的固体,也就是集“超流体+固体”特性于一身的物质。简单来说就是超固体既有晶体态中原子规则排布的特征,又可以像超流体一样无摩擦流动。在凝聚态物理学中,超固体是具有超流体特性的空间有序材料。超固体是一种晶体材料,其原子排列结构规则且重复,也能够永远流动而不损失任何动能。尽管它

SpringMVC之自定义注解

目录一、Java注解1.1注解简介1.2注解分类1.3JDK基本注解1.4JDK元注解1.5自定义注解1.5.1标记注解1.5.2元数据注解1.6如何自定义注解二、自定义注解的基本案例2.1案例一(获取类、方法以及属性上的注解)2.1.1@Ingerited的使用2.2案例二(获取类属性上的注解属性值)2.3案例三(获

全球变暖问题(floodfill 处理联通块问题)

全球变暖问题文章目录全球变暖问题前言题目描述题目分析边界问题的考虑岛屿是否被淹没判断:如何寻找联通块:代码预告前言之前我们介绍了bfs算法在二维,三维地图中的应用,现在我们接续进行拓展,解锁floodfill算法,准确的来说是用bfs算法解决联通块问题。后续还会更新bfs算法有关内容,喜欢的小伙伴可以点个关注啦。题目描

数据结构和算法之快速排序

快速排序是一种基于分治法的排序算法。它通过不断地将数组分成较小的子数组,并按照递归的方式对每个子数组进行排序,最终将整个数组排序。#mermaid-svg-Za26UnuASULzGzsM{font-family:"trebuchetms",verdana,arial,sans-serif;font-size:16px

Vue路由与nodejs环境搭建

一.路由什么是路由什么是SPA路由的思路及实现实例建立一个HTML来编写路由测试结果​编辑二.nodejs环境什么是node.jsnpm是什么node.js的下载一.路由什么是路由路由(Routing)是指根据不同的URL地址,将用户导航到不同的页面或视图的过程。在前端开发中,特别是在单页面应用(SPA)中,路由起着至

热文推荐