Spring的IOC详解

IOC=控制反转(思想),把「创建对象、管理依赖」的控制权,从程序员手里,交给容器。容器解决了"对象管理"问题。不用自己 new 对象了

DI=依赖注入(实现IOC的手段),实现分为:构造器注入、setter()注入、接口注入三种,将对象需要的依赖传递给它,而不是让对象自行创建依赖

Spring官方推荐

Spring团队明确推荐使用构造器注入,主要原因:

  1. 确保依赖不可变:final关键字保证
  2. 确保对象完全初始化:构造完成后就可使用
  3. 避免空指针异常:依赖不可能为null
  4. 更好的可测试性:构造时就确定所有依赖
  5. 非强制加final关键字,官方推荐规范加final

注入方式示例

1、构造器注入

复制代码
/**
 * 用户 Controller
 */
@RestController
@RequestMapping("/")
public class UserController {

    // 1. 定义 private final
    private final UserService userService;

    /**
     * 构造器注入
     * 2. 只有一个构造器,Spring 自动注入,不用 @Autowired
     */
    public UserController(UserService userService) {
        this.userService = userService;
    }
}

lombok简化版@RequiredArgsConstructor:自动给所有 final 字段生成构造器\Spring 自动识别注入\ 不需要写任何构造器

复制代码
/**
 * 用户 Controller
 */
@RestController
@RequestMapping("/")
@RequiredArgsConstructor    // 关键
public class UserController {

    private final UserService userService;
}

2、setter()注入

Setter 注入是通过类的 setter 方法来注入依赖,需要配合@Autowired注解使用(Spring 4.3+ 也可使用@RequiredArgsConstructor,但手动写 setter 更直观)。

复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;

@Controller // 补充Controller注解,确保Spring能扫描到该类
public class UserController {

    // 1. 仅定义私有变量,去掉final(setter注入不能用final,因为final变量必须在构造器/声明时赋值)
    private UserService userService;

    /**
     * Setter注入
     * 2. 通过setter方法注入依赖,必须加@Autowired注解
     */
    @Autowired
    public void setUserService(UserService userService) {
        this.userService = userService;
    }
}

3、字段注入

字段注入是直接在类的成员变量上标注@Autowired,无需构造器或 setter 方法,是最简洁的写法,但也是 Spring 官方不推荐的写法(不利于单元测试)。

复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;

@Controller
public class UserController {

    // 1. 直接在字段上标注@Autowired,无需构造器/setter
    @Autowired
    private UserService userService; // 同样不能用final
}

可选依赖和必选依赖判断标准(核心看「这个依赖缺失时,你的类 / 功能是否还能正常工作」------ 这不是技术规则,而是业务逻辑和功能设计决定)

|------|-------------------------|---------------------|
| 类型 | 核心特征 | 典型场景 |
| 必选依赖 | 依赖缺失 → 类无法初始化 / 功能完全不可用 | 数据库 DAO、核心业务服务、配置中心 |
| 可选依赖 | 依赖缺失 → 类能正常工作,仅部分功能降级 | 缓存服务、日志增强、消息通知 |


IOC不是管理所有对象,只管【Bean】

✅ Spring IoC 容器 管:

被你加上这些注解的类:

  • @Controller------ 控制层
  • @Service------ 业务层
  • @Repository------ 数据层(Dao、Mapper)
  • @Component ------ 通用组件(工具类、配置类、自定义组件等)
  • @Configuration 配置类
  • @Bean 方法返回的对象

这些叫 Bean,Spring 负责创建、注入、销毁。


❌ Spring IoC 容器 不管:

  • 实体类(User、Order、Product...)
  • VO / DTO / BO / POJO
  • 普通工具类对象
  • 方法内部临时创建的对象
  • new 出来的一切普通对象

这些不是 Bean,Spring 不负责,必须你自己 new。


相关推荐
VidDown15 小时前
Webhook 调试器:让第三方回调“原形毕露”
java·开发语言·javascript·编辑器·postman
折哥的程序人生 · 物流技术专研15 小时前
Java 23 种设计模式:从踩坑到精通 | 原型模式 —— 克隆对象,深拷贝与浅拷贝的坑你踩过吗?
java·设计模式·架构·原型模式·单一职责原则
装不满的克莱因瓶16 小时前
基于 OpenResty 扩展开发实现动态服务注册与发现能力
java·开发语言·架构·openresty
程序员小羊!16 小时前
06Java 异常机制与常用类
java
weixin_5231853216 小时前
Java基础知识总结(四):引用数据类型与参数传递机制
java·开发语言·python
Nayxxu16 小时前
Claude API 生产稳定性设计:超时、降级、备用模型和告警怎么做
开发语言·php
王cb16 小时前
WinRT Server and Client c#
开发语言·c#
Selina K17 小时前
C中日历时间转换
c语言·开发语言
怪我冷i17 小时前
zig语言学习笔记——heap-memory
开发语言·golang·zig
宸津-代码粉碎机17 小时前
Spring AI企业级实战|从RAG优化到Agent多工具调度
java·大数据·人工智能·后端·python·spring