探究Spring的单例设计模式--单例Bean

ops/2024/10/21 10:02:21/

Spring的单例设计模式

在Spring框架中,单例设计模式是一种常见且重要的设计模式,主要用于确保在应用程序的生命周期中仅创建一个特定的Bean实例

一、什么是单例设计模式

单例设计模式是一种创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。这种模式的核心在于控制实例的创建,避免了因为多次创建对象导致的资源浪费。单例模式通常用于以下场景:

  1. 节约资源:在需要大量创建相同对象时,使用单例可以显著减少内存消耗。

  2. 全局访问:单例模式提供了一个全局访问点,使得在任何地方都能轻松获取到该实例,避免了需要通过参数传递对象的复杂性。

  3. 协调操作:在某些情况下,多个对象需要共享状态或资源,单例模式能够确保所有对象都访问到相同的数据。

在Spring框架中,单例Bean是默认的Bean作用域。开发者定义的Bean如果没有显式指定作用域,Spring容器将自动将其视为单例。Spring通过内置的机制管理单例Bean的创建和访问,确保应用在运行时始终只有一个实例。

这种设计模式使得Spring的开发者可以专注于业务逻辑,而不必担心实例化和管理对象的细节。Spring通过依赖注入(DI)机制,将单例Bean的实例注入到需要它的类中,提供了简单而强大的方式来共享对象。

二、Spring中的单例Bean

在Spring中,当我们定义一个Bean时,如果没有指定作用域,它默认是单例的。以下是一个简单的示例:

java">import org.springframework.stereotype.Component;@Component
public class MySingletonBean {public MySingletonBean() {System.out.println("MySingletonBean instance created");}public void doSomething() {System.out.println("Doing something...");}
}

在这个示例中,MySingletonBean是一个单例Bean。当Spring容器启动时,它会创建这个Bean的唯一实例。

三、单例Bean的实现原理

在Spring框架中,单例Bean的实现依赖于Spring容器的管理机制:

1.单例Bean的创建

Spring容器负责Bean的实例化、配置和生命周期管理。单例Bean的创建主要在AbstractAutowireCapableBeanFactory类中进行。具体的创建流程如下:

  1. Bean定义的注册:首先,在Spring容器启动时,所有的Bean定义(包括其元数据)会被注册到DefaultListableBeanFactory中。

  2. Bean的实例化:当一个Bean被请求时,Spring会检查是否已经存在该Bean的实例。对于单例Bean,Spring在第一次请求时会创建实例,并将其存储在缓存中。

2.源码解析

下面是Spring源码中与单例Bean管理相关的几个重要部分:

3.单例Bean的生命周期管理

除了实例化,Spring还负责单例Bean的生命周期管理,包括:

初始化:Bean被实例化后,可以执行初始化方法,例如通过@PostConstruct注解或实现InitializingBean接口。

销毁:在容器关闭时,Spring会调用单例Bean的销毁方法,例如通过@PreDestroy注解或实现DisposableBean接口。

  1. Bean的创建逻辑:在AbstractAutowireCapableBeanFactory中,doGetBean方法负责获取Bean实例。以下是部分代码片段:

    java">@Override
    protected <T> T doGetBean(String name, Class<T> requiredType, Object[] args) {// 检查缓存Object bean = getSingleton(name);if (bean != null) {return (T) bean;}// 如果缓存中不存在,则创建新实例// 省略 Bean 实例化逻辑bean = createBean(name, mbd, args);// 将创建的 Bean 存入缓存registerSingleton(name, bean);return (T) bean;
    }​
    public Object getSingleton(String beanName, ObjectFactory<?> singletonFactory) {Assert.notNull(beanName, "'beanName' must not be null");synchronized (this.singletonObjects) {// 检查缓存中是否存在实例Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject == null) {//...省略了很多代码try {singletonObject = singletonFactory.getObject();}//...省略了很多代码// 如果实例对象在不存在,我们注册到单例注册表中。addSingleton(beanName, singletonObject);}return (singletonObject != NULL_OBJECT ? singletonObject : null);}}//将对象添加到单例注册表protected void addSingleton(String beanName, Object singletonObject) {synchronized (this.singletonObjects) {this.singletonObjects.put(beanName, (singletonObject != null ? singletonObject : NULL_OBJECT));}}
    }​
    
    • getSingleton(name):检查是否存在该Bean的实例。
    • createBean(name, mbd, args):创建新的Bean实例。
    • registerSingleton(name, bean):将新创建的Bean注册到单例缓存中。
  2. 单例缓存:Spring使用singletonObjects集合来缓存单例Bean实例,具体在DefaultSingletonBeanRegistry中定义:

    java">protected final Map<String, Object> singletonObjects = new ConcurrentHashMap<>(256);
    
    1. 当一个单例Bean被创建后,Spring会将其实例存储在这个singletonObjects映射中,以便后续请求时直接返回。
  3. 线程安全:Spring确保单例Bean在多线程环境中的安全性。对于单例Bean的创建,通常会在关键部分添加同步机制,以防止多个线程同时创建实例。例如,在创建单例Bean时,Spring会使用synchronized关键字来保证线程安全。

四、总结

Spring的单例设计模式通过确保Bean在整个应用程序中只有一个实例,提供了高效的资源管理和简化的访问方式。理解单例Bean的实现原理及其在多线程环境中的安全性,对于开发高效的Spring应用至关重要。


http://www.ppmy.cn/ops/118352.html

相关文章

Apollo Planning2.0决策规划算法代码详细解析 (3):PlanningComponent框架介绍

Apollo Planning 2.0的框架更新涉及多个方面&#xff0c;这些更新旨在提升自动驾驶系统的灵活性、可扩展性和性能。 以下是Apollo Planning 2.0 的框架图&#xff1a; 其中&#xff0c;Apollo的PlanningComponent在自动驾驶系统中扮演着至关重要的角色。其主要作用可以归纳为以…

Pinia从安装到使用

什么是Pinia 添加Pinia到vue项目 使用Pinia实现计数器案例 counter.js import {defineStore} from "pinia"; import {ref} from "vue";export const useCounterStore defineStore(coutner,()>{//定义数据&#xff08;state&#xff09;const count r…

DualGS:高效人体体积视频渲染技术,实现复杂4D数字人表演的实时播放引言

随着虚拟现实(VR)和增强现实(AR)技术的发展,对高质量、低延迟的人体体积视频的需求日益增长。传统的视频压缩和渲染方法在处理复杂的4D人体动作时往往面临性能瓶颈。为了解决这一问题,研究人员开发了一种名为DualGS的新型高效人体体积视频渲染技术。本文将详细介绍DualGS…

Vue ElemetUI table实现双击修改编辑某个内容

1、使用cell-dblclick事件&#xff0c;当双击时触发事件 <el-table cell-dblclick"handleCellDblClick" 2、单元格设置 主要重点为判断双击时切换input框&#xff0c;然后绑定ref&#xff0c;设置失去焦点时触发点方法&#xff0c;与按enter键触发点方法 <…

shell编程:dirname命令获取路径详解

dirname 命令是 Unix/Linux 系统中用于获取给定路径中的目录部分的命令。它不会改变当前工作目录&#xff0c;而是简单地分析提供的路径字符串&#xff0c;并输出该路径的目录部分。 基本用法 命令格式&#xff1a; dirname [OPTION]... PATH...OPTION&#xff1a;通常&#x…

如何使用ssm实现线上旅游体验系统+vue

TOC ssm691线上旅游体验系统vue 绪论 课题背景 身处网络时代&#xff0c;随着网络系统体系发展的不断成熟和完善&#xff0c;人们的生活也随之发生了很大的变化。目前&#xff0c;人们在追求较高物质生活的同时&#xff0c;也在想着如何使自身的精神内涵得到提升&#xff0…

通过WinCC在ARMxy边缘计算网关上实现智能运维

随着信息技术与工业生产的深度融合&#xff0c;智能化运维成为提升企业竞争力的关键因素之一。ARMxy系列的ARM嵌入式计算机BL340系列凭借其高性能、高灵活性和广泛的适用性&#xff0c;为实现工业现场的智能运维提供了坚实的硬件基础。 1. 概述 ARMxy BL340系列是专为工业应用…

如何批量获取安卓设备上所有应用的桌面图标

文章目录 概述一、准备工作1.1 安装 ADB安装 ADB&#xff1a;1.2 获取设备上的应用包名1.3 使用 ADB 导出 APK 文件 二、提取桌面图标流程2.1 反编译 APK 文件2.2 解析 AndroidManifest.xml2.3 搜索图标资源文件 三、 编写 Python 自动化工具3.1 代码结构3.2 运行脚本 四、错误…