深入理解Spring源码之自动装配

简介: 深入理解Spring源码之自动装配

自动装配;

        Spring利用依赖注入(DI),完成对IOC容器中中各个组件的依赖关系赋值;

 

1)、@Autowired:自动注入:

        1)、默认优先按照类型去容器中找对应的组件:applicationContext.getBean(BookDao.class);找到就赋值
        2)、如果找到多个相同类型的组件,再将属性的名称作为组件的id去容器中查找
                            applicationContext.getBean("bookDao")
        3)、@Qualifier("bookDao"):使用@Qualifier指定需要装配的组件的id,而不是使用属性名
        4)、自动装配默认一定要将属性赋值好,没有就会报错;
             可以使用@Autowired(required=false);
        5)、@Primary:让Spring进行自动装配的时候,默认使用首选的bean;
                 也可以继续使用@Qualifier指定需要装配的bean的名字

     

BookService{
                    @Autowired
                    BookDao  bookDao;
                }

 2)、Spring还支持使用@Resource(JSR250)和@Inject(JSR330)[java规范的注解]

         @Resource:

             可以和@Autowired一样实现自动装配功能;默认是按照组件名称进行装配的;

             没有能支持@Primary功能没有支持@Autowired(reqiured=false);

         @Inject:

             需要导入javax.inject的包,和Autowired的功能一样。没有required=false的功能;

  @Autowired:Spring定义的; @Resource、@Inject都是java规范

     

 AutowiredAnnotationBeanPostProcessor:解析完成自动装配功能;        

 

 3)、 @Autowired:构造器,参数,方法,属性;都是从容器中获取参数组件的值

         1)、[标注在方法位置]:@Bean+方法参数;参数从容器中获取;默认不写@Autowired效果是一样的;都能自动装配

         2)、[标在构造器上]:如果组件只有一个有参构造器,这个有参构造器的@Autowired可以省略,参数位置的组件还是可以自动从容器中获取

        3)、放在参数位置:

 

        4)、自定义组件想要使用Spring容器底层的一些组件(ApplicationContext,BeanFactory,xxx);

                 自定义组件实现xxxAware;在创建对象的时候,会调用接口规定的方法注入相关组件;Aware;

                 把Spring底层一些组件注入到自定义的Bean中;

                 xxxAware:功能使用xxxProcessor;

                 ApplicationContextAware==》ApplicationContextAwareProcessor;

 

举例查看ApplicationContextAware的完整初始化过程:

public class IOCTest_Autowired {
  @Test
  public void test01(){
    AnnotationConfigApplicationContext applicationContext = new 
            AnnotationConfigApplicationContext(MainConifgOfAutowired.class);
  }
}
package com.atguigu.bean;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanNameAware;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.EmbeddedValueResolverAware;
import org.springframework.stereotype.Component;
import org.springframework.util.StringValueResolver;
@Component
public class AwareBean implements ApplicationContextAware,BeanNameAware,EmbeddedValueResolverAware {
  private ApplicationContext applicationContext;
  @Override
  public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
    // TODO Auto-generated method stub
    System.out.println("传入的ioc:"+applicationContext);
    this.applicationContext = applicationContext;
  }
  @Override
  public void setBeanName(String name) {
    // TODO Auto-generated method stub
    System.out.println("当前bean的名字:"+name);
  }
  @Override
  public void setEmbeddedValueResolver(StringValueResolver resolver) {
    // TODO Auto-generated method stub
    String resolveStringValue = resolver.resolveStringValue("你好 ${os.name} 我是 #{20*18}");
    System.out.println("解析的字符串:"+resolveStringValue);
  }
}

     断点查看调用栈如下:

-->AnnotationConfigApplicationContext refresh();
   -->finishBeanFactoryInitialization(beanFactory); 
      -->beanFactory.preInstantiateSingletons()
        -->DefaultListableBeanFactory getBean(beanName)
          -->AbstractBeanFactory doGetBean(name, null, null, false);
            -->AbstractBeanFactory getSingleton(beanName, new ObjectFactory<Object>() 
              --> DefaultSingletonBeanRegistry singletonFactory.getObject()
                 -->AbstractBeanFactory createBean(beanName, mbd, args)
                   -->AbstractAutowireCapableBeanFactory doCreateBean(beanName, mbdToUse, args)    
                     -->AbstractAutowireCapableBeanFactory initializeBean(beanName, exposedObject, mbd)   
                       -->AbstractAutowireCapableBeanFactory applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName)
                         -->AbstractAutowireCapableBeanFactory beanProcessor.postProcessBeforeInitialization(result, beanName);
                            -->ApplicationContextAwareProcessor    invokeAwareInterfaces(bean);
                              -->ApplicationContextAwareProcessor ((ApplicationContextAware) bean).setApplicationContext(this.applicationContext);
                                -->AwareBean setApplicationContext

 

打开ApplicationContextAwareProcessor的源码发现它实现了BeanPostProcessor前置类在postProcessBeforeInitialization方法里调用invokeAwareInterfaces最后一步时设置spring容器

private void invokeAwareInterfaces(Object bean) {
  if (bean instanceof Aware) {
    if (bean instanceof EnvironmentAware) {
        ((EnvironmentAware) bean).setEnvironment(this.applicationContext.getEnvironment());
    }
    if (bean instanceof EmbeddedValueResolverAware) {
        ((EmbeddedValueResolverAware) bean).setEmbeddedValueResolver(this.embeddedValueResolver);
    }
    if (bean instanceof ResourceLoaderAware) {
        ((ResourceLoaderAware) bean).setResourceLoader(this.applicationContext);
    }
    if (bean instanceof ApplicationEventPublisherAware) {
        ((ApplicationEventPublisherAware) bean).setApplicationEventPublisher(this.applicationContext);
    }
    if (bean instanceof MessageSourceAware) {
        ((MessageSourceAware) bean).setMessageSource(this.applicationContext);
    }
    if (bean instanceof ApplicationContextAware) {
        ((ApplicationContextAware) bean).setApplicationContext(this.applicationContext);
    }
  }
}

 


目录
相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
1月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
18天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
38 2
|
1月前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
64 9
|
3月前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
316 24
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
3月前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
228 24
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
3月前
|
XML 缓存 Java
手写Spring源码(简化版)
Spring包下的类、手写@ComponentScan注解、@Component注解、@Autowired注解、@Scope注解、手写BeanDefinition、BeanNameAware、InitializingBean、BeanPostProcessor 、手写AnnotationConfigApplicationContext
手写Spring源码(简化版)
|
2月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
148 5
|
2月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
2月前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
146 9