前端视角 Java Web 入门手册 5.5:真实世界 Web 开发——控制反转与 @Autowired

高层模块管理底层模块

在传统的编程模式中,高层模块需要直接创建和管理低层模块的实例,我们称之为高层模块控制底层模块的生命周期,这种控制关系导致模块之间高度耦合

java 复制代码
public class UserService {
    private UserRepository userRepository = new UserRepository(); // 高层直接控制低层

    public void createUser(User user) {
        userRepository.save(user);
    }
}

UserService 直接创建并依赖于 UserRepository 的具体实现,这样 UserRepository 无法直接替换为不同的实现,比如从本地存储切换到远程存储,需要同时修改 UserService 的代码。

同时这样的写法也带来测试的复杂性,为 UserService 编写单元测试,但由于它直接创建了 UserRepository 的实例,无法轻松地替换为一个 Mock 对象

java 复制代码
public class UserServiceTest {
    @Test
    public void testCreateUser() {
        UserService userService = new UserService();
        User user = new User("John Doe", "john@example.com");
        userService.createUser(user);
        // 无法验证或模拟 UserRepository 的行为
    }
}

控制反转与依赖注入

控制反转是指将对象的创建和依赖管理交由外部容器(如 Spring IoC 容器)来实现,从而降低模块之间的耦合度

  • 所有组件的实例化和生命周期管理由 IoC 容器处理
  • 上层模块不再自行创建下层模块的实例,而是通过构造器、Setter 方法或字段注入的方式获得依赖。
java 复制代码
// 下层模块接口
public interface UserRepository {
    void save(User user);
}

// 下层模块实现
@Repository
public class UserRepositoryImpl implements UserRepository {
    @Override
    public void save(User user) {
        // 保存用户到数据库
    }
}

// 上层模块
@Service
public class UserService {
    private final UserRepository userRepository;
    
    @Autowired // Spring 自动注入依赖
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
    
    public void createUser(User user) {
        userRepository.save(user);
    }
}

在上述示例中:

  • UserRepository 接口定义了保存用户的方法,而 UserRepositoryImpl 提供了具体实现,UserService 依赖 UserRepository 接口,而不依赖其实现。
  • UserService 通过构造器注入的方式获得 UserRepository 的实例,由 Spring IoC 容器负责注入具体实现 UserRepositoryImpl。

控制反转(IoC) 是设计原则,依赖注入(DI) 是实现控制反转的具体方式。

这样 UserRepositoryImpl 的修改不影响 UserService 的代码,并且在单元测试中,可以使用 Mockito 替换 UserRepository 为一个模拟对象,实现隔离测试

java 复制代码
@ExtendWith(MockitoExtension.class)
public class UserServiceTest {
    
    @Mock
    private UserRepository userRepository;
    
    @InjectMocks
    private UserService userService;
    
    @Test
    public void testCreateUser() {
        User user = new User("John Doe", "john@example.com");
        
        userService.createUser(user);
        
        verify(userRepository, times(1)).save(user);
    }
}

@Autowired 做了什么

@Autowired 是 Spring Framework 提供的一个核心注解,主要用于实现依赖注入。@Autowired 注解用于标注在类的构造器、Setter 方法或字段上,表示 Spring 容器需要为这些位置提供相应的依赖对象。Spring 容器在启动时会扫描应用上下文中的所有 Bean,自动完成依赖的注入。

构造器注入

将依赖通过构造器参数传递,这是推荐的注入方式,因为它确保依赖在对象创建时就被提供,符合不可变性原则。

java 复制代码
@Service
public class UserService {

    private final UserRepository userRepository;

    @Autowired // 从 Spring 4.3 开始如果只有一个构造器,可以省略 @Autowired
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    // 业务逻辑方法
}

Setter 注入

通过 Setter 方法注入依赖对象,适用于可选依赖或需要在运行时修改依赖的场景。

java 复制代码
@Service
public class UserService {

    private UserRepository userRepository;

    @Autowired
    public void setUserRepository(UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    // 业务逻辑方法
}

字段注入

直接在字段上注入依赖,是最简洁的注入方式,但不推荐用于生产代码,因为它不利于测试和违反了依赖注入的设计原则。

java 复制代码
@Service
public class UserService {

    @Autowired
    private UserRepository userRepository;

    // 业务逻辑方法
}

注入优先级

当存在多个符合条件的 Bean 时,Spring 无法确定注入哪个 Bean,这时需要结合使用 @Qualifier@Primary 来明确指定。

可以在注入时候使用 @Qualifier 指定要注入的 Bean 的名称

java 复制代码
@Service
public class UserService {

    private final UserRepository userRepository;

    @Autowired 
    public UserService(@Qualifier("userRepositoryImpl") UserRepository userRepository) {
        this.userRepository = userRepository;
    }

    // 业务逻辑方法
}

也可以使用@Primary 注解用于标记某个 Bean 为默认 Bean,当存在多个 Bean 时,优先选择被标记的 Bean。

java 复制代码
@Repository
@Primary
public class UserRepositoryImpl implements UserRepository {
    // 实现细节
}

@Repository
public class AlternativeUserRepository implements UserRepository {
    // 其他实现
}
相关推荐
星浩AI5 分钟前
手把手带你在 Windows 安装 Hermess Agent,并接入飞书 [喂饭级教程含踩坑经验]
人工智能·后端·agent
神奇小汤圆11 分钟前
Spring Boot 入门:Java 生态最流行的应用开发框架介绍
后端
龙月13 分钟前
Gitlab迁移与升级技术方案
前端·后端
张小洛18 分钟前
Spring 常用类深度剖析(工具篇 04):CollectionUtils 与 Stream API 的对比与融合
java·后端·spring·spring工具类·spring utils·spring 类解析
kunge201336 分钟前
UBUNTU Claude Code 报错 claude native binary not installed
后端
暮年40 分钟前
Java Map并发-Hashtable
后端
一 乐41 分钟前
房产租赁管理|基于springboot + vue房产租赁管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·房产租赁管理系统
未秃头的程序猿42 分钟前
从零到一:深入浅出分布式锁原理与Spring Boot实战(Redis + ZooKeeper)
spring boot·分布式·后端
Soofjan43 分钟前
MySQL(3.2):索引应用与优化
后端
黑牛儿44 分钟前
PHP 8.3性能暴涨实测|对比8.2,接口响应提速30%,配置无需大幅修改
android·开发语言·后端·php