核心内容摘要
[特殊字符]️Qwen2.5-VL-7B-Instruct效果实测:快递单多角度拍摄图→统一结构化运单信息抽取
【Spring】超详细一篇文章让你完全理解 Spring IoC 和 DI大家好我是 Grok。
今天我们来聊聊 Spring 框架的核心灵魂——IoCInversion of Control控制反转和DIDependency Injection依赖注入。
如果你是 Spring 小白或者想彻底搞懂这两个概念这篇文章绝对够用我们从基础原理入手逐步深入到实际代码、配置和高级玩法保证逻辑清晰、易记。
为什么说“超详细”因为我会用通俗比喻像“厨师 vs 餐厅老板”来解释抽象概念。
表格对比快速区分传统方式 vs Spring 方式。
代码示例Maven Java 简单项目复制就能跑。
步骤拆解从 XML 配置到注解、Java 配置全覆盖。
常见坑避免初学者踩雷。
假设你有 Java 基础类、接口、Maven没问题就开始吧
分基础概念——IoC 和 DI 到底是什么
1 IoC控制反转Inversion of Control传统编程你程序员像个“厨师”自己控制一切——创建对象、调用方法、管理依赖。
比如你要一个“汽车”对象就自己new Car()然后自己组装轮子、引擎。
IoC 的核心思想把“控制权”反转给别人Spring 容器。
你不再自己new对象而是告诉 Spring“我需要一个汽车”Spring 像“餐厅老板”一样自动帮你组装好、注入依赖、甚至管理生命周期。
为什么需要 IoC传统方式耦合太紧改一个类就得改一堆地方。
IoC 解耦让代码更灵活、可测试、可扩展。
IoC 的好处3 大关键词解耦模块间不直接依赖改动不连锁反应。
复用对象像“乐高积木”随时组装。
管理Spring 统一管理对象创建、销毁、作用域。
IoC 是“思想”DI 是“实现方式”。
IoC 就像“外卖平台”DI 就像“骑手送餐”。
2 DI依赖注入Dependency InjectionDI 是 IoC 的具体实现Spring 不是让你自己找依赖而是“注入”给你。
依赖什么一个对象需要另一个对象才能工作。
比如Car需要Engine和Wheel。
注入怎么做Spring 通过配置XML/注解/Java自动把依赖“塞”进你的对象。
DI 的 3 种方式按注入时机分方式描述优点/缺点示例场景构造函数注入通过构造方法注入依赖强制依赖对象创建时就注入测试友好核心依赖如数据库连接Setter 注入通过 Setter 方法注入灵活可选依赖但可能注入不全可选配置如日志级别接口注入通过接口方法注入较少用动态但复杂遗留系统集成DI 让你的代码从“主动拉取”依赖变成“被动接收”。
3 IoC vs DI 的关系别混淆IoC 是大概念控制反转包括 DI 和其他如 AOP。
DI 是 IoC 的子集具体怎么反转控制通过注入依赖。
Spring 的 IoC 容器ApplicationContext或BeanFactory管理所有 Bean对象。
一句话IoC 是“老板指挥”DI 是“老板给你发工具”。
分Spring 中的 IoC 和 DI 实战Spring 版本假设用 Spring Boot
7 或 Spring Framework
x2026 年主流。
先建个 Maven 项目!-- pom.xml 核心依赖 --dependenciesdependencygroupIdorg.springframework/groupIdartifactIdspring-context/artifactIdversion
6.
0/version!-- 最新版自行替换 --/dependency/dependencies
1 配置方式 1XML 配置经典但老派XML 是 Spring 最早的配置方式适合理解底层。
步骤拆解定义 BeanXML 文件中声明对象。
注入依赖用property或constructor-arg。
获取容器ClassPathXmlApplicationContext加载 XML。
示例代码接口和类publicinterfaceEngine{voidstart();}publicclassPetrolEngineimplementsEngine{publicvoidstart(){System.out.println(Petrol Engine started);}}publicclassCar{privateEngineengine;// 依赖// Setter 注入publicvoidsetEngine(Engineengine){this.engineengine;}publicvoiddrive(){engine.start();System.out.println(Car is driving);}}XML 配置applicationContext.xml?xml version
0 encodingUTF-8?beansxmlnshttp://www.springframework.org/schema/beansxmlns:xsihttp://www.w
org/2001/XMLSchema-instancexsi:schemaLocationhttp://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd!-- 定义 Bean --beanidpetrolEngineclasscom.example.PetrolEngine/beanidcarclasscom.example.Car!-- DISetter 注入 --propertynameenginerefpetrolEngine//bean/beans测试publicclassMain{publicstaticvoidmain(String[]args){ApplicationContextcontextnewClassPathXmlApplicationContext(applicationContext.xml);Carcar(Car)context.getBean(car);// IoC 容器提供car.drive();// 输出: Petrol Engine started \n Car is driving}}构造函数注入版 XML替换propertybeanidcarclasscom.example.Carconstructor-argrefpetrolEngine//beanCar 类加构造public Car(Engine engine) { this.engine engine; }
2 配置方式 2注解配置现代主流简单从 Spring
5 开始注解取代 XML。
Spring Boot 默认用这个。
关键注解注解作用示例Component标记类为 Bean通用Component public class Car {}Service业务层 BeanServiceRepositoryDAO 层 BeanRepositoryControllerWeb 层 BeanControllerAutowired自动注入依赖byType默认Autowired private Engine engine;Qualifier指定 Bean 名当多个同类型时Qualifier(“petrol”)Configuration配置类代替 XMLConfiguration public class AppConfig {}Bean方法返回 BeanBean public Engine engine() {}ComponentScan扫描包自动发现 BeanComponentScan(“com.example”)示例代码注解版类ComponentpublicclassPetrolEngineimplementsEngine{publicvoidstart(){System.out.println(Petrol Engine started);}}ComponentpublicclassCar{AutowiredprivateEngineengine;// DI 注入publicvoiddrive(){engine.start();System.out.println(Car is driving);}}配置类可选如果不扫描默认包ConfigurationComponentScan(com.example)// 扫描包publicclassAppConfig{}测试publicclassMain{publicstaticvoidmain(String[]args){ApplicationContextcontextnewAnnotationConfigApplicationContext(AppConfig.class);Carcarcontext.getBean(Car.class);car.drive();}}自动装配模式Autowired 的底层byType按类型匹配默认。
byName按 Bean 名匹配用 Qualifier。
3 配置方式 3Java 配置纯代码零 XML/注解污染适合大型项目配置集中。
示例基于
2 的类去掉 ComponentConfigurationpublicclassAppConfig{BeanpublicEngineengine(){returnnewPetrolEngine();}BeanpublicCarcar(){CarcnewCar();c.setEngine(engine());// 手动 DIreturnc;}}测试同上new AnnotationConfigApplicationContext(AppConfig.class)。
分高级玩法——Bean 管理和优化
1 Bean 作用域Scope默认 Singleton单例但可以改。
Scope描述注解/配置singleton全局唯一实例默认prototype每次 getBean 都新创建Scope(“prototype”)request每个 HTTP 请求一个Scope(“request”) (Web)session每个会话一个Scope(“session”)示例Scope(prototype) public class Car {}
2 Bean 生命周期IoC 容器管理从创建到销毁的全过程实例化new。
属性注入DI。
初始化PostConstruct 或 init-method。
使用。
销毁PreDestroy 或 destroy-method。
自定义publicclassCar{PostConstructpublicvoidinit(){System.out.println(Init);}PreDestroypublicvoiddestroy(){System.out.println(Destroy);}}
3 自动装配AutowiringEnableAutoConfigurationSpring Boot 用自动配置常见 Bean如 DataSource。
问题多个同类型 Bean用 Primary 标记首选或 Qualifier 指定。
分
常见问题 踩坑指南问题/坑原因解决NoSuchBeanDefinitionExceptionBean 未定义或未扫描检查 Component / Bean / 扫描包UnsatisfiedDependencyException依赖未注入检查 Autowired / 类型匹配循环依赖A 依赖 BB 依赖 A用 Setter 注入 / Lazy 延迟XML vs 注解性能XML 解析慢优先注解 / Java 配置测试时注入失败上下文未加载用 SpringBootTest调试技巧日志级别调到 DEBUG看 Spring 如何装配 Bean。
分
总结 扩展学习核心回顾IoC 是反转控制DI 是注入依赖。
Spring 通过容器ApplicationContext管理 Bean实现解耦和灵活。
为什么 Spring 牛因为 IoC/DI 让大型项目像“积木”一样搭建易维护。
下一步学 AOP切面编程然后 Spring Boot简化版 Spring。
练习建议建个小项目注入 Service 到 Controller模拟用户注册。
如果你有具体代码问题、或想看 Spring Boot 版示例继续问我比如“帮我写个带数据库的 DI 示例”