news 2026/2/28 10:27:35

Spring IOC/DI 核心知识

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Spring IOC/DI 核心知识

一、核心容器架构

1. 容器层级关系

  • BeanFactory:根容器,提供基础的IoC功能,采用延迟加载策略
  • ApplicationContext:子容器,继承BeanFactory,增加AOP、消息源、事件发布等企业级功能
  • WebApplicationContext:专为Web应用设计的上下文,支持Request/Session作用域

2. 容器实现类

类型加载来源使用场景
ClassPathXmlApplicationContext类路径下的XML文件传统XML配置项目
AnnotationConfigApplicationContextJava配置类注解驱动项目
FileSystemXmlApplicationContext文件系统绝对路径特殊文件位置需求

二、IOC与DI核心概念

IOC(Inversion of Control - 控制反转)

本质:将对象的创建、存储、管理权限从程序员转移给Spring容器

  • 控制:创建对象的能力
  • 反转:将创建权交给Spring,我们只需获取使用

DI(Dependency Injection - 依赖注入)

本质:在容器创建对象后,自动为其属性赋值(注入依赖)

  • 作用:解耦对象间的依赖关系
  • 方式:XML配置、注解、自动注入

个人理解口诀:IOC创建对象,DI给属性赋值


三、XML配置详解

1. Bean定义基础

<beanid="userService"class="com.luo.service.impl.UserServiceImpl"scope="singleton"lazy-init="false"autowire="byType"><!-- 属性注入 --><propertyname="userDao"ref="userDao"/><propertyname="name"value="小罗"/></bean><beanid="userDao"class="com.luo.dao.impl.UserDaoImpl"/>

关键属性

  • id:Bean唯一标识(默认首字母小写类名)
  • class:类全路径
  • scope:作用域(singleton/prototype)
  • lazy-init:懒加载(默认false,启动时创建)
  • autowire:自动注入模式

2. 依赖注入三种方式

  1. Setter注入:通过<property>标签调用setter方法
  2. 构造器注入:通过<constructor-arg>标签
  3. 自动注入autowire="byType|byName|constructor"

四、Bean生命周期(重点背诵)

完整流程图

1. 解析配置 → BeanDefinition(元数据) 2. 实例化 → 构造函数创建对象 3. 属性注入 → DI(依赖注入) 4. 初始化 → 三种方式 ├─ @PostConstruct ├─ InitializingBean.afterPropertiesSet() └─ init-method指定 5. 使用 → getBean()返回实例 6. 销毁 → 三种方式 ├─ @PreDestroy ├─ DisposableBean.destroy() └─ destroy-method指定

单例 vs 多例

特性单例(singleton)多例(prototype)
创建时机容器启动时创建(默认)getBean()时创建
存储位置singletonObjects缓存集合不缓存,每次都新建
内存占用占用空间换时间效率每次创建新实例
线程安全需注意共享资源天然线程安全

懒加载lazy-init="true"使单例Bean在第一次获取时才创建


五、BeanDefinition vs singletonObjects

对象类型存储内容创建时机存储位置
BeanDefinitionBean元数据(类名、作用域、属性、依赖等)容器启动解析配置时BeanFactory元数据区
singletonObjects已实例化、注入、初始化的Bean对象单例Bean实例化后单例缓存池

关系:BeanDefinition是模板,singletonObjects是最终产品


六、扩展接口(高阶背诵)

1. BeanFactoryPostProcessor(Bean工厂后处理器)

执行时机:所有Bean实例化之前,只执行一次
作用:修改Bean定义、动态注册Bean、解析占位符
典型应用PropertySourcesPlaceholderConfigurer解析${}

2. BeanPostProcessor(Bean后处理器)

执行时机:Bean实例化后、初始化前后各执行一次
作用:AOP代理创建、依赖增强、性能监控
Spring AOP本质:通过BeanPostProcessor在初始化后创建代理对象

执行顺序

实例化 → 属性注入 → postProcessBeforeInitialization → 初始化 → postProcessAfterInitialization → 就绪可用

七、FactoryBean机制(复杂对象创建)

使用场景

  • 对象没有无参构造函数
  • 创建代理对象
  • 第三方框架整合
  • 动态创建复杂对象

实现步骤

  1. 创建类实现FactoryBean<T>接口
  2. 重写三个方法:
    • getObject():返回实际对象(会加入容器)
    • getObjectType():返回对象类型
    • isSingleton():是否单例(默认true)
  3. 配置Bean:<bean id="user" class="UserFactoryBean"/>

注意:容器中Bean的ID是getObject()返回的对象标识,而非FactoryBean本身


八、外部属性文件引入

<!-- 1. 引入properties文件 --><context:property-placeholderlocation="classpath:jdbc.properties"/><!-- 2. 使用占位符 --><beanid="dataSource"class="com.alibaba.druid.pool.DruidDataSource"><propertyname="driverClassName"value="${jdbc.driver}"/><propertyname="url"value="${jdbc.url}"/><propertyname="username"value="${jdbc.username}"/><propertyname="password"value="${jdbc.password}"/></bean>

作用:实现配置与代码分离,方便环境切换


九、Bean获取方式对比

// 1. 通过ID获取(不推荐,强依赖字符串)Objectbean=ctx.getBean("userService");// 2. 通过ID+类型(特定场景)UserServiceservice=ctx.getBean("userService",UserService.class);// 3. 通过类型(推荐,类型安全)UserServiceservice=ctx.getBean(UserService.class);

注意:同一类型多个Bean时,必须使用ID+类型方式


十、自动注入(Autowire)

模式规则适用场景
byType按类型匹配(默认)容器中该类型Bean唯一
byName按属性名匹配ID同类型多个Bean时
constructor按构造器参数类型构造器注入时

注解等价物

  • @Autowired=autowire="byType"
  • @Resource(name="xxx")=autowire="byName"

十一、核心记忆口诀

容器篇

  • 根容器:BeanFactory
  • 子容器:ApplicationContext
  • 加载方式:ClassPathXml、AnnotationConfig

IOC/DI篇

  • IOC:控制反转,创建对象交给Spring
  • DI:依赖注入,属性赋值自动化
  • 配置:XML(property)、注解(@Value)

生命周期篇

  • 四阶段:实例化 → 注入 → 初始化 → 使用
  • 两后置:BeanPostProcessor前后拦截
  • 一工厂:BeanFactoryPostProcessor前置修改

作用域篇

  • 单例:启动创建,缓存共享(默认)
  • 多例:获取创建,每次新建
  • 懒加载:延迟单例创建时间

十二、面试高频问答

Q1: BeanFactory和ApplicationContext区别?
A: BeanFactory是基础容器,延迟加载;ApplicationContext是增强版,立即加载,支持更多企业级功能。

Q2: 单例Bean线程安全吗?
A: Spring不保证线程安全,需开发者自己保证。多例Bean天然线程安全。

Q3: @Autowired和@Resource区别?
A: @Autowired按类型注入(默认),@Resource按名称注入(默认)。

Q4: BeanPostProcessor作用?
A: 在Bean初始化前后执行,Spring AOP就是基于此实现的动态代理。

Q5: FactoryBean和BeanFactory区别?
A: BeanFactory是IoC容器工厂,FactoryBean是创建特殊Bean的工厂Bean。


版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/18 3:14:50

20年IT老兵亲授:Claudecode AI编程工具从安装到实战!

简介 本文是一位近20年IT经验的老IT人分享的AI辅助编程工具Claudecode使用教程。文章详细介绍了Claudecode的安装、配置方法&#xff0c;并通过开发一个H5五子棋小游戏展示了其实际应用。Claudecode能够根据自然语言描述自动生成高质量代码&#xff0c;非常适合编程能力较弱但有…

作者头像 李华
网站建设 2026/2/26 15:16:58

48、使用 Visual Studio 创建 Windows 窗体应用程序

使用 Visual Studio 创建 Windows 窗体应用程序 在 Windows 窗体应用程序开发中,虽然手动编写代码有助于理解模型的简单性,但通常我们会使用 Visual Studio IDE 以快速应用程序开发(RAD)的方式来设计窗体。下面我们通过一个示例来详细了解这个过程。 创建空的 Windows 窗…

作者头像 李华
网站建设 2026/2/26 10:16:01

51、Windows Forms 自定义用户控件与布局技巧

Windows Forms 自定义用户控件与布局技巧 1. 自定义用户控件 在设计应用程序时,有时表单会变得复杂,包含大量控件。一般应避免这种情况,因为单个表单上控件过多意味着需要有经验丰富的用户。不过,表单上常存在功能相关的控件组,支持它们的代码大多可与表单上其他控件的代…

作者头像 李华
网站建设 2026/2/25 4:39:02

56、数据集合操作与事务处理全解析

数据集合操作与事务处理全解析 在数据处理过程中,我们常常需要对数据集合进行搜索、合并等操作,同时也会涉及到事务处理以保证数据的一致性和完整性。下面将详细介绍这些操作的方法和技巧。 1. 搜索数据集合 当数据被加载到数据集合(DataSet)后,我们可能需要从中搜索和…

作者头像 李华
网站建设 2026/2/27 20:15:58

2、深入解析Flex与Bison:从原理到应用

深入解析Flex与Bison:从原理到应用 1. Flex与Bison的起源 Flex和Bison是在编程领域广泛应用的工具,它们的起源可以追溯到上世纪。Bison源自yacc,yacc是由贝尔实验室的Stephen C. Johnson在1975 - 1978年间编写的解析器生成器。它结合了D. E. Knuth解析工作的坚实理论基础和…

作者头像 李华
网站建设 2026/2/26 22:38:20

3、深入探索 Flex:正则表达式、文件 I/O 与输入输出管理

深入探索 Flex:正则表达式、文件 I/O 与输入输出管理 1. 正则表达式基础 正则表达式是 Flex 扫描器的核心,它使用一种丰富的元语言来描述模式。Flex 的正则表达式语言本质上是 POSIX 扩展正则表达式。以下是一些具有特殊含义的字符及其功能: | 字符 | 功能 | | ---- | -…

作者头像 李华