Spring依赖注入底层原理详解

 更新时间:2023年09月14日 08:58:04   作者:这堆干货有点猛  
这篇文章主要介绍了Spring依赖注入底层原理详解,  依赖注入是一种设计模式,它将对象之间的依赖关系从代码中移除,并由容器来管理这些依赖关系,依赖注入的主要目的是降低代码的耦合度,使代码更加灵活和可维护,需要的朋友可以参考下

Spring 框架作为 Java 开发中最流行的框架之一,其核心特性之一就是依赖注入(Dependency Injection,DI)。

在Spring中,依赖注入是通过 IOC 容器(Inversion of Control,控制反转)来实现的。

本文将详细介绍Spring的依赖注入底层原理,并提供源码示例。

什么是依赖注入

依赖注入是一种设计模式,它将对象之间的依赖关系从代码中移除,并由容器来管理这些依赖关系。依赖注入的主要目的是降低代码的耦合度,使代码更加灵活和可维护。

在 Java 中,依赖通常是通过构造函数或者 Setter 方法来注入的。使用依赖注入,我们可以将对象的创建和依赖关系的管理分离开来,从而使得代码更加容易测试和维护。

实现原理

Spring的依赖注入是通过 IOC 容器来实现的。在Spring中,IOC 容器负责创建和管理对象,以及管理对象之间的依赖关系。

IOC 容器

IOC 容器是指用于管理对象和依赖关系的容器。Spring提供了多种 IOC 容器实现,包括 BeanFactory 和 ApplicationContext 等。

BeanFactory 是 Spring 中最基本的 IOC 容器,它提供了基本的 IOC 功能。

ApplicationContext 则是BeanFactory的扩展,它提供了更多的功能,如事件发布、国际化支持、AOP等。

在 Spring 中,BeanFactory 和 ApplicationContext 都是通过反射来实例化对象,并通过依赖注入来管理对象之间的依赖关系。

Bean定义

在 Spring 中,每个被管理的对象都需要有一个对应的 Bean 定义。Bean定义是一个元数据,它描述了一个 Bean 的类型、属性、依赖关系等信息。

Bean 定义通常是通过XML配置文件、 Java 配置类或者注解来定义的。下面是一个使用 XML 配置文件定义 Bean 的示例:

<bean id="userService" class="com.example.UserService">
    <property name="userRepository" ref="userRepository"/>
</bean>
<bean id="userRepository" class="com.example.UserRepositoryImpl"/>

在上面的示例中,我们定义了一个名为 userService的Bean,它的类型是com.example.UserService。它依赖于另一个名为userRepository的Bean,类型是com.example.UserRepositoryImpl。

依赖注入

在 Spring 中,依赖注入是通过反射来实现的。当 IOC 容器创建 Bean 时,它会检查 Bean 定义中所声明的依赖关系,并尝试通过反射来注入这些依赖关系。

依赖注入通常分为三种方式:构造函数注入、Setter 方法注入和字段注入。

构造函数注入

构造函数注入是最常见的依赖注入方式。在 Spring 中,我们可以通过构造函数来注入 Bean 的依赖关系。当 IOC 容器创建 Bean 时,它会检查 Bean 定义中所声明的构造函数,并尝试通过反射来调用这个构造函数,并将依赖关系作为参数传递进去。

下面是一个使用构造函数注入的示例:

public class UserService {
    private final UserRepository userRepository;
    public UserService(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
    // ...
}

在上面的示例中,我们定义了一个 UserService 类,并将 UserRepository 依赖关系通过构造函数注入进去。

Setter方法注入

Setter 方法注入是另一种常见的依赖注入方式。我们可以通过 Setter 方法来注入Bean的依赖关系。当 IOC 容器创建Bean时,它会检查 Bean 定义中所声明的 Setter 方法,并尝试通过反射来调用这个 Setter 方法,并将依赖关系作为参数传递进去。

下面是一个使用Setter方法注入的示例:

public class UserService {
    private UserRepository userRepository;
    public void setUserRepository(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
    // ...
}

在上面的示例中,我们定义了一个 UserService 类,并将 UserRepository 依赖关系通过 Setter 方法注入进去。

字段注入

字段注入是一种不太常见的依赖注入方式。我们可以通过字段来注入Bean的依赖关系。当 IOC 容器创建 Bean 时,它会尝试通过反射来注入这些字段。

下面是一个使用字段注入的示例:

public class UserService {
    @Autowired
    private UserRepository userRepository;
    // ...
}

在上面的示例中,我们使用了 @Autowired 注解来将 UserRepository 依赖关系注入到 userService 对象中的 userRepository 字段中。

生命周期回调

在Spring中,Bean生命周期包括四个阶段:实例化、属性赋值、初始化、销毁。在这些阶段中,Spring 提供了多个回调方法,以便我们在 Bean 的生命周期中进行一些自定义操作。

下面是一些常用的Bean生命周期回调方法:

  • 实例化:Bean 实例化之后,Spring 会调用 BeanPostProcessor 的 postProcessBeforeInitialization 方法。
  • 属性赋值:在Bean实例化之后,Spring 会将 Bean 定义中所声明的属性值赋值给 Bean 对象。
  • 初始化:在 Bean 属性赋值之后,Spring 会调用 InitializingBeanafterPropertiesSet 方法或者 @Bean 注解的 initMethod 方法。
  • 销毁:在 IOC 容器关闭时,Spring 会调用 DisposableBean 的 destroy 方法或者 @Bean 注解的 destroyMethod方 法。

下面是一个实现 InitializingBean 和 DisposableBean 接口的示例:

public class UserService implements InitializingBean, DisposableBean {
    private UserRepository userRepository;
    public void setUserRepository(UserRepository userRepository) {
        this.userRepository = userRepository;
    }
    public void afterPropertiesSet() throws Exception {
        // 在Bean属性赋值之后,执行一些初始化操作
    }
    public void destroy() throws Exception {
        // 在 IOC 容器关闭时,执行一些销毁操作
    }
    // ...
}

注解

在 Spring 中,我们可以使用注解来简化 Bean 定义和依赖注入的过程。Spring 提供了多个注解来实现依赖注入和生命周期回调等功能。

下面是一些常用的 Spring 注解:

  • @Component:用于标记一个类为 Bean。
  • @Autowired:用于标记一个字段、构造函数或者 Setter 方法需要进行依赖注入。
  • @Qualifier:当存在多个相同类型的 Bean 时,用于指定依赖注入的具体实现。
  • @Value:用于注入一个常量值。
  • @PostConstruct:用于标记一个方法为 Bean 初始化方法。
  • @PreDestroy:用于标记一个方法为 Bean 销毁方法。

下面是一个使用注解的示例:

@Component
public class UserService {
    @Autowired
    private UserRepository userRepository;
    @PostConstruct
    public void init() {
        // 在Bean属性赋值之后,执行一些初始化操作
    }
    @PreDestroy
    public void destroy() {
        // 在 IOC 容器关闭时,执行一些销毁操作
    }
    // ...
}

源码示例

下面是一个使用Spring的依赖注入功能的示例:

@Component
public class UserService {
    @Autowired
    private UserRepository userRepository;
    public void save(User user) {
        userRepository.save(user);
    }
}
@Component
public class UserRepositoryImpl implements UserRepository {
    public void save(User user) {
        // 保存用户信息
    }
}
@Configuration
public class AppConfig {
    @Bean
    public UserService userService() {
        return new UserService();
    }
    @Bean
    public UserRepository userRepository() {
        return new UserRepositoryImpl();
    }
}
public class Main {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
        UserService userService = context.getBean(UserService.class);
        User user = new User();
        userService.save(user);
    }
}

在上面的示例中,我们使用了 @Component 注解标记了 UserService 和 UserRepositoryImpl 两个类为 Bean。在 AppConfig 类中,我们使用 @Bean 注解来定义了 UserRepository 和 UserService 两个 Bean。

在 Main 类中,我们使用 AnnotationConfigApplicationContext 来创建 IOC 容器,并通过依赖注入来获取 UserService 对象,并调用它的 save 方法。

总结

本文详细介绍了 Spring 的依赖注入底层原理,并提供了源码示例。

通过本文的学习,我们可以更好地理解 Spring 的依赖注入机制,以及如何在实际开发中使用它来降低代码的耦合度,使代码更加灵活和可维护。

到此这篇关于Spring依赖注入底层原理详解的文章就介绍到这了,更多相关Spring依赖注入内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • mybatis模糊查询like语句该如何写

    mybatis模糊查询like语句该如何写

    MyBatis模糊查询通常使用LIKE关键字,结合concat函数拼接通配符%实现,在MyBatis配置文件中,通过#{keyword}传递参数,生成带有通配符的查询语句,MyBatis-Plus中,通过LambdaQueryWrapper类和like方法构建模糊查询条件,简化查询操作
    2024-09-09
  • Idea上传、拉取、更新项目到gitee的实现

    Idea上传、拉取、更新项目到gitee的实现

    该文章介绍了将Idea项目上传至Gitee的步骤,包括创建本地仓库、添加到缓冲区、提交到本地仓库、推送至远程仓库等以及从Gitee拉取项目的方法,整个过程涉及到创建远程仓库、项目上传、更新项目等和提交等内容
    2026-04-04
  • Springboot中redis使用lettuce连接池经常连接超时问题分析及解决

    Springboot中redis使用lettuce连接池经常连接超时问题分析及解决

    文章主要介绍了在Spring Boot项目中,使用Lettuce作为Redis客户端时遇到的问题——连接在一段时间后会自动掉线,问题的根源在于Lettuce的空闲连接处理机制,文章提供了Spring Boot 2.3及以上版本的解决方案,即通过配置解决,或者排除Lettuce并采用Jedis
    2025-11-11
  • SpringBoot设置接口超时的方法小结

    SpringBoot设置接口超时的方法小结

    这篇文章主要介绍了SpringBoot设置接口超时的方法小结,包括配置文件,config配置类及相关示例代码,代码简单易懂,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-09-09
  • Java中JS引擎实现的一句话木马

    Java中JS引擎实现的一句话木马

    这篇文章主要为大家介绍了如何利用Java中JS引擎实现的一句话木马,有需要的朋友可以借鉴参考下,希望能够有所帮助祝大家多多进步
    2022-02-02
  • SpringMvc/SpringBoot HTTP通信加解密的实现

    SpringMvc/SpringBoot HTTP通信加解密的实现

    这篇文章主要介绍了SpringMvc/SpringBoot HTTP通信加解密的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • idea导入项目框架的详细操作方法

    idea导入项目框架的详细操作方法

    大家使用idea开发工具时经常会需要导入项目框架,纠结该怎么操作呢,今天小编给大家分享一篇图文教程,帮助大家解决idea导入项目框架的问题,感兴趣的朋友一起看看吧
    2021-05-05
  • 使用spring的restTemplate注意点

    使用spring的restTemplate注意点

    这篇文章主要介绍了使用spring的restTemplate注意点,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • 如何使用JaCoCo分析java单元测试覆盖率

    如何使用JaCoCo分析java单元测试覆盖率

    在做单元测试时,代码覆盖率常常被拿来作为衡量测试好坏的指标,甚至,用代码覆盖率来考核测试任务完成情况,比如,代码覆盖率必须达到80%或 90%。于是乎,测试人员费尽心思设计案例覆盖代码。下面我们来学习一下吧
    2019-06-06
  • Java实战之敏感词过滤器

    Java实战之敏感词过滤器

    这篇文章主要介绍了Java实战之敏感词过滤器,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04

最新评论