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

news2024/9/21 19:00:17

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 小结

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

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

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

相关文章

JAVA高级技术入门(单元测试,反射,注解,动态代理)

JAVA高级技术入门(单元测试,反射,注解,动态代理) 一、Junit单元测试二、反射1.认识反射,获取类概念:快速入门:获取Class对象的三种方式 2.1获取类的构造器2.2获取类的构造器的作用&a…

计算机系统概述之计算机的发展历程

计算机系统概述之计算机的发展历程 计算机的发展历程计算机系统硬件的发展微处理器的发展 软件的发展CAD/ CAM/CIMS的简单介绍 思维导图总结 计算机的发展历程 计算机系统 计算机系统由硬件和软件组成。 硬件:指的是计算机实体,如:主机&#…

【深度学习】 Python 和 NumPy 系列教程(廿四):Matplotlib详解:2、3d绘图类型(10)3D箱线图(3D Box Plot)

目录 一、前言 二、实验环境 三、Matplotlib详解 1、2d绘图类型 2、3d绘图类型 0. 设置中文字体 1. 3D线框图(3D Line Plot) 2. 3D散点图(3D Scatter Plot) 3. 3D条形图(3D Bar Plot) 4. 3D曲面图…

动态规划——01背包

背包问题经典资料背包九讲,可以上网查一下相关资料。 下面的资料来自代码随想录和自己的一些个人理解,如有需要可以跳转代码随想录进行学习:代码随想录 (programmercarl.com) 背包一共分为01背包,完全背包,多重背包&am…

leetcode:70. 爬楼梯

一、题目 函数原型:int climbStairs(int n) 二、思路 此题运用递归思想。当只有1个台阶,那么只有1种方法爬到楼顶——跨一个台阶;当有2个台阶时,有2种方法爬到楼顶——跨一个台阶跨两次或直接跨两个台阶。当有3个台阶或更多台阶时…

实现客户端pineline的思路

背景: redis集群不支持客户端的mget操作,但是业务上对这个redis集群的批量操作的需求一直都在,所以有各种客户端实现了各式各样的pineline实现,本文就记录下我们公司的实现方式 pineline实现思路 1.pineline要快 pineline之所以快是因为可…

深度学习训练过程可视化工具

1.深度学习网络结构画图工具 地址:https://cbovar.github.io/ConvNetDraw/ 2.caffe可视化工具 输入:caffe配置文件 输出:网络结构 地址:http://ethereon.github.io/netscope/#/editor 3.深度学习可视化工具Visual DL Visual D…

数据变换:数据挖掘的准备工作之一

⭐️⭐️⭐️⭐️⭐️欢迎来到我的博客⭐️⭐️⭐️⭐️⭐️ 🐴作者:秋无之地 🐴简介:CSDN爬虫、后端、大数据领域创作者。目前从事python爬虫、后端和大数据等相关工作,主要擅长领域有:爬虫、后端、大数据…

【毕设选题】flink大数据淘宝用户行为数据实时分析与可视化

文章目录 0 前言1、环境准备1.1 flink 下载相关 jar 包1.2 生成 kafka 数据1.3 开发前的三个小 tip 2、flink-sql 客户端编写运行 sql2.1 创建 kafka 数据源表2.2 指标统计:每小时成交量2.2.1 创建 es 结果表, 存放每小时的成交量2.2.2 执行 sql &#x…

【python手写算法】numpy实现简易神经网络和反向传播算法【1】

import numpy as npdef dense(A,W):Znp.matmul(A,W)#矩阵乘法return 1/(1np.exp(-Z))if __name__ __main__:leanring_rate100Anp.array([[200.0,17.0]])# Wnp.array([[1,-3,5],# [-2,4,-6]])# bnp.array([[-1,1,2]])W1 np.array([[0., -10, 4],[-1,3,2]])W2np.ar…

数学建模__非线性规划Python实现

使用到的是scipy库 线性规划指的是目标模型均为线性,除此以外的都是非线性规划,使用scipy提供的方法对该类问题进行求解。 from scipy.optimize import minimize import numpy as np#定义目标函数 def fun(args):a,b,c,d argsv lambda x: (ax[0])/ (b…

免费和开源的机器翻译软件LibreTranslate

什么是 LibreTranslate ? LibreTranslate 免费开源机器翻译 API,完全自托管。与其他 API 不同,它不依赖于 Google 或 Azure 等专有提供商来执行翻译。它的翻译引擎由开源 Argos Translate 库提供支持。 这个软件在 2022 年 3 月的时候折腾过&…

Vue 3的革命性新特性:深入了解Composition API

🌷🍁 博主猫头虎(🐅🐾)带您 Go to New World✨🍁 🦄 博客首页——🐅🐾猫头虎的博客🎐 🐳 《面试题大全专栏》 🦕 文章图文…

数据结构与算法——8.二分查找

这篇文章我们来讲一下数据结构与算法中的二分查找 目录 1.介绍 1.1背景介绍 1.2算法介绍 2.实现 3.几个问题 4.算法改进 4.1左闭右开版 4.2 平衡版 4.3 Leftmost版 4.4 Leftmost返回 i 版 5.小结 1.介绍 首先,我们来介绍一下二分查找 1.1背景介绍 需…

Marmof:AI写作助手文章内容生成器工具

【产品介绍】 名称 Marmof 具体描述 Marmof是一个AI驱动的写作平台,可以帮助你快速创建原创、无抄袭的网站、邮件、广告和博客内容。每月提供5000字的免费额度,永久有效。拥有超过49种强大的工具,可以为各种平台生成吸引人的内容…

界面组件DevExpress WinForms v23.1亮点 - 全新升级HTML CSS模板

DevExpress WinForms拥有180组件和UI库,能为Windows Forms平台创建具有影响力的业务解决方案。DevExpress WinForms能完美构建流畅、美观且易于使用的应用程序,无论是Office风格的界面,还是分析处理大批量的业务数据,它都能轻松胜…

Jupyter杂症二:目录不显示或展示不全

文章目录 序言一、问题表象二、解决及思考 有些问题,解法还真的是莫名其妙,就如同拍了一下电视机就好了。。。。。难道是在拍的时候,我们传了内力给了计算机? 序言 一、问题表象 二、解决及思考 用鼠标按住左键,在空…

vim,emacs,verilog-mode这几个到底是啥关系?

vim:不多说了被各类coder誉为地表最强最好用的编辑器;gvim,gui vim的意思; emacs:也是一个编辑器,类似vscode; vim在使用的时候为了增强其功能,有好多好多插件,都是以.…

Brother CNC联网数采集和远程控制

兄弟CNC IP地址设定参考:https://www.sohu.com/a/544461221_121353733没有能力写代码的兄弟可以提前下载好网络调试助手NetAssist,这样就不用写代码来测试连接CNC了。 以上是网络调试助手抓取CNC的产出命令,结果有多个行string需要自行解析&…

应用商店备案登记流程解析

​ 引言: 随着智能手机的普及和移动互联网的发展,移动应用程序(App)已成为人们日常生活中不可或缺的一部分。在开发一个App之后,开发者需要将其上传到应用商店进行审核和上架。然而,在上架之前&#xff0…