Spring 注解驱动开发

放假期间,看一些Spring相关的注解,加深一下印象,

鸣谢:尚硅谷视频,此总结来源于视频内容

1. Spring 组件注册


给容器中中注册组件
1、包扫描+组件标注注解(@Component、@Service、@Controller、@Repository,主要是自己写的类
2、@Bean[导入的第三方包里面的组件]
3、@Import[快速给容器中导入一个组件]
         1、Import(类名),容器中就会自动注册这个组件,id默认是组件的全名
         2、ImportSelector:返回需要导入的组件的全类名的数组
         3、ImportBeanDefinitionRegistrar:手动注册bean
4、使用Spring提供的FactoryBean(工厂bean)
         1、默认获取到的是工厂bean调用getObject创建的对象
         2、要获取到bean本身,需要给id前面加个&标识
@Conditional({Condition}):按照一定的条件判断,满足条件给容器中注册bean
@Scope
prototype:多例的 ioc容器启动并不会去调用方法创建对象在容器中,而是每次获取时才会调用方法创建对象
singleton:单例的(默认值) ioc容器启动会调用方法创建对象放到ioc容器中,以后每次获取就是从容器中拿

1.1 Spring 组件注册注册时过滤条件


@ComponentScan value:指定要扫描的包

excludeFilters=Filter[]:指定扫描包的时候按照什么规则排除哪些组件

includeFilters=Filter[]:指定扫描包的时候要包含哪些组件,需将useDefaultFilters置false

FilterType.ANNOTATION:按照注解

FilterType.ASSIGNABLE_TYPE:按照指定的类型

FilterType.REGEX:使用正则指定

FilterType.CUSTOM:使用自定义规则

2. Spring 生命周期

 bean的生命周期
      bean创建--初始化--销毁的过程
 可以自定义初始化和销毁方法,容器在bean进行到当前声明周期的时候来调用我们自定义的初始化和销毁方法
 构造(对象创建)
       单实例:在容器启动时创建
       多实例:每次获取时创建
 初始化:对象创建时
 销毁:
      单实例:关闭容器时
      多实例:容器不会管理这个bean,容器不会调用销毁方法、

 遍历得到容器中所有的BeanPostProcessor;挨个执行beforeInitialization
 一旦返回null,跳出for循环,不会执行后面的BeanPostProcess.postProcessors
 BeanPostProcessor的大致执行流程
 populateBean(beanName, mbd, instanceWrapper);给bean进行属性赋值
 initializeBean{
   applyBeanPostProcessorsBeforeInitialization//for循环得到全部beanPost
   invokeInitMethods(beanName, wrappedBean, mbd);//初始化方法
   applyBeanPostProcessorsAfterInitialization//for循环得到全部beanPost
}


 1)指定初始化和销毁方法 在@Bean注解里指定init方法和destroy方法
 2)通过让bean实现InitializingBean(定义初始化逻辑),DisposableBean(定义销毁逻辑)接口
 3)可以使用JSR250
      1、@PostConstructor:在bean创建完成并且属性赋值完成,来执行初始化方法
      2、@PreDestroy:在容器销毁bean之前通知我们进行清理工作
 4)BeanPostProcessor【接口】:bean的后置处理器
      在bean初始化前后进行一些处理工作
          postProcessBeforeInitialization:在初始化之前进行一些工作
          对象初始化
          postProcessAfterInitialization:在初始化之后进行一些工作
      Spring底层对BeanPostProcess接口的使用
      bean赋值,注入其它组件,@AutoWired,生命周期注解功能 @Async xxx都是通过BeanPostProcess进行完成的
      ApplicationContextAwareProcessor会在实现了ApplicationContextAware接口的bean里面
      通过((ApplicationContextAware) bean).setApplicationContext(this.applicationContext);将容器注入到bean中

      InitDestroyAnnotationBeanPostProcessor会对@PostConstructor和@PreDestroy注解进行解析来达到容器的初始化和销毁方法的执行

2.1 属性赋值

@Value赋值
1、基本数值
2、可以写SpEL;#{}
3、可以写${}取出配置文件【properties】的值(在运行环境变量里面的值)
使用@PropertySource读取外部配置文件中的k/v保存到运行的环境变量中;加载完外部的配置文件后使用${}取出配置文件的值

3. Spring 依赖注入,自动在装配


 自动装配;
      Spring利用依赖注入(DI),完成对IOC容器中各个组件的依赖关系赋值;
      1、@AutoWired:自动注入【Spring定义的】
          1、默认按照类型去容器中找对应的组件applicationContext.getBean(BookService.class),找到就赋值
          2、如果找到相同类型的组件,再将属性的名称作为组件的id去容器中查找applicationContext.getBean("bookDao")
          3、@Qualifier("bookDao"):使用该注解来指定需要装配的组件的id,而不是使用属性名
          4、自动装配默认一定要将属性赋值好,没有就会报错,可通过在Autowire的注解中将required=false来使该配置设置为非必需
          5、@Primary:让Spring进行自动装配的时候,默认使用首选的bean,也可以继续使用@Qualifier来指定需要装配的bean
          BookService{
              @Autowired
              BookDao bookDao;
          }
      2、Spring还支持使用@Resource(JSR250)和@Inject(JSR330)【java规范】
          1、@Resource:
              可以和@Autowired一样实现自动装配功能;默认是按照组件名称进行装配的;没有能支持@Primary的功能以及@Autowired(required=false)的功能
          2、@Inject(需要导入依赖):
              导入javax.inject的包,和Autowired的功能一样,没有required=false的功能

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

      3、@Autowired:构造器,参数,方法,属性
        1)标注在方法位置标注在方法,Spring容器创建当前对象,就会调用方法,完成赋值,方法使用的参数,自定义类型的值从ioc容器中获取,@Bean标注的方法创建对象的时候,方法参数的值默认从ioc容器中获取,默认不写Autowired,效果是一样的
        2)标注在构造器位置默认加在ioc容器中的组件,容器启动会调用无参构造器创建对象,再进行初始化赋值等操作。标注在构造器上可以默认调用该方法,方法中用的参数同样从ioc容器中获取,如果容器只有一个有参构造器,这个有参构造器的Autowired可以省略,参数位置的组件还是可以自动从容器中获取
        3)标注在参数位置从ioc容器中获取参数组件的值

      4、自定义组件想要使用Spring容器底层的一些组件(ApplicationContext,BeanFactory,xxx);
           自定义组件需要实现xxxAware接口;在创建对象的时候会调用接口规定的方法注入相关组件;
           把Spring底层的一些组件注入到自定义的bean中
           xxxAware的功能都是使用xxxAwareProcessor处理的



 Profile:
      Spring为我们提供的可以根据当前环境,动态的激活和切换一系列组件的功能;

 开发环境、测试环境、生产环境
 数据源(/A)(/B)(/C)

 @Profile:指定组件在哪个环境的情况下才能被注册到容器中,不指定,任何环境下都能注册

 1、加了环境标识的bean,只有这个环境被激活的时候才能注册到容器中,默认是default环境
 2、写在配置类上,只有是指定的环境的时候,整个配置类里面的所有配置才能生效
 3、没有标注环境标识的bean在任何环境下都加载

 运行时如何指定运行环境:
  1、命令行参数,通过在虚拟机参数位置指定-Dspring.profiles.active=xxx来指定运行环境,标注了该环境的bean会被配置进容器中
  2、程序内指定:
          1、创建一个applicationContext
          2、设置需要激活的环境,applicationContext.getEnvironment().setActiveProfiles("");
          3、注册主配置类,applicationContext.register(xxx.class)
          4、启动刷新容器,applicationContext.refresh();

3. Spring AOP

3.1 AOP使用


 AOP:【动态代理】
      指在程序运行期间动态的将某段代码切入到指定方法指定位置进行运行的编程模式
  1、导入aop模块
  2、定义一个业务逻辑类(MathCalculator),在业务逻辑运行的时候将日志进行打印(方法之前,方法运行结束,方法出现异常等)
  3、定义一个日志切面类(LogAspect),切面类里面的方法需要动态感知MathCalculator.div运行到哪里,然后执行
      通知方法:
              前置通知(@Before):logStart:在目标方法(div)运行之前运行 参数列表传入joinPoint可获取到方法的相关属性,且该参数必须放在第一个参数,否则无法识别
              后置通知(@After):logEnd:在目标方法(div)运行之后运行,无论方法正常结束还是异常结束
              返回通知(@AfterReturning(returning可以指定封装返回值的参数)):logReturn:在目标方法(div)正常返回之后运行 
              异常通知(@AfterThrowing):logException:在目标方法(div)出现异常以后运行
              环绕通知(@Around):动态代理,手动推进目标方法运行(joinPoint.proceed())
  4、给切面类的目标方法标注何时何地运行(通知注解)
  5、将切面类和目标和业务逻辑类(目标方法所在类)都加入到容器中;
  6、告诉Spring哪个类是切面类(给切面类上加一个注解@Aspect)
  [7]、需要给配置类加一个@EnableAspectJAutoProxy【开启基于注解的aop模式】
      在spring中很多的@EnableXXX;

 三步:
      1、将业务逻辑组件和切面类都加入到容器中,告诉spring哪个是切面类(@Aspect)
      2、在切面类上的每一个通知方法标注通知注解,告诉spring何时何地运行(切入点表达式)
      3、开启基于注解的aop模式@EnableAspectJAutoProxy

3.2 AOP的原理


 AOP原理:【看给容器中注册了什么组件,这个组件什么时候工作,包括这个组件工作时候的功能】      @EnableAspectJAutoProxy
 1、@EnableAspectJAutoProxy是什么?
      @Import(AspectJAutoProxyRegistrar.class):给容器导入AspectJAutoProxyRegistrar.class
          利用AspectJAutoProxyRegistrar自定义给容器中注册bean
          第一步给容器中注册了"internalAutoProxyCreator"=AnnotationAwareAspectJAutoProxyCreator的bean(BeanDefinition),即bean的定义信息
 2、AnnotationAwareAspectJAutoProxyCreator extends
      AnnotationAwareAspectJAutoProxyCreator extends
          AspectJAwareAdvisorAutoProxyCreator extends
              AbstractAdvisorAutoProxyCreator  extends
                  AbstractAutoProxyCreator extends ProxyProcessorSupport implements SmartInstantiationAwareBeanPostProcessor, BeanFactoryAware
关注后置处理器(在bean初始化完成前后做事情)(SmartInstantiationAwareBeanPostProcessor)、自动       装配beanFactory(BeanFactoryAware)
    装备beanFactory的逻辑
    AbstractAutoProxyCreator.setBeanFactory()
    后置处理器的逻辑
    AbstractAutoProxyCreator.postProcessBeforeInstantiation()
    AbstractAdvisorAutoProxyCreator.setBeanFactory()重写了父类方法,会在方法里调一个initBeanFactory()方法
    AspectJAwareAdvisorAutoProxyCreator
    AnnotationAwareAspectJAutoProxyCreator.initBeanFactory(),即调用setBeanFactory时会调用这里的initBeanFactory()方法
 流程:
      1)传入主配置类,创建ioc容器
      2)注册配置类,调用refresh()刷新容器
      3)registerBeanPostProcessors(beanFactory);注册bean的后置处理器来方便拦截bean的创建
              1、先获取ioc容器中已经定义了的需要创建对象的所有BeanPostProcessor beanFactory.getBeanNamesForType();
              2、给容器中加别的BeanPostProcessor beanFactory.addBeanPostProcessor();
              3、对实现了PriorityOrdered接口和Ordered接口以及其它的BeanPostProcessor以进行分类
              4、优先注册实现了PriorityOrdered接口的BeanPostProcessor ,其次注册实现了Ordered接口的BeanPostProcessor,最后注册其它的
              5、AnnotationAwareAspectJAutoProxyCreator实现了Ordered接口,注册这个BeanPostProcessor,实际上就是创建这个对象,保存在容器中
                     如何创建名为internalAutoProxyCreator的AnnotationAwareAspectJAutoProxyCreator对象
                     doCreateBean():
                     1、创建bean的实例
                     2、populateBean();给bean的各种属性赋值
                     3、initializeBean();初始化bean
                              1、invokeAwareMethods();处理Aware接口的方法回掉(AnnotationAwareAspectJAutoProxyCreator实现了BeanFactoryAware接口,会在这里将beanFactory回调给该bean)
                              2、applyBeanPostProcessorsBeforeInitialization();执行后置处理器的BeforeInitialization
                              3、invokeInitMethods();执行初始化方法
                              4、applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);;执行后置处理器的AfterInitialization
                     4、AnnotationAwareAspectJAutoProxyCreator创建成功,并且通过beanFactory创建了aspectJAdvisorFactory,再用aspectJAdvisorFactory创建了aspectJAdvisorsBuilder
              6、把AnnotationAwareAspectJAutoProxyCreator这个bean注册到BeanFactory中,beanFactory.addBeanPostProcessor(postProcessor);
 ===============以上是创建和注册AnnotationAwareAspectJAutoProxyCreator的过程====================================================================
         AnnotationAwareAspectJAutoProxyCreator是InstantiationAwareBeanPostProcessor类型的后置处理器
      4)finishBeanFactoryInitialization(beanFactory);完成beanFactory初始化工作;创建剩余的单实例bean
          1、遍历获取容器中所有的Bean,依次创建对象
                getBean()->doGetBean()->getSingleton()
          2、创建bean【AnnotationAwareAspectJAutoProxyCreator会在所有bean创建之前有一个拦截,会调用 postProcessBeforeInstantiation方法】
              1、先从缓存中获取当前bean,如果能获取到,说明bean是之前被创建过的,直接使用,否则再创建,只要创建好的bean都会被缓存起来
              2、createBean();创建bean AnnotationAwareAspectJAutoProxyCreator会在任何bean创建之前先尝试返回bean的实例
                  【BeanPostProcessor是在Bean对象创建完成初始化前后调用的】
                  【InstantiationAwareBeanPostProcessor是在创建bean实例之前先尝试用后置处理器返回对象】
                  1、resolveBeforeInstantiation();解析BeforeInstantiation,希望后置处理器在此能返回一个代理对象,如果能返回就使用该对象;
                          1)后置处理器先尝试返回对象 bean = applyBeanPostProcessorsBeforeInstantiation();
                                  applyBeanPostProcessorsBeforeInstantiation()方法中会拿到所有后置处理器,
                                  判断如果是InstantiationAwareBeanPostProcessor类型的,就执行postProcessBeforeInstantiation方法
                                          1)判断当前bean是否在advisedBeans(保存了所有需要增强的bean)中
                                          2)判断当前bean是否是基础类型的Advice、PointCut、Advisor、AopInfrastructureBean,或者是否是切面(@Aspect)
                                          3)是否需要跳过
                                              1、获取候选的增强器(切面里面的通知方法)【List<Advisor> candidateAdvisors】,
                                                  每一个封装的通知方法的增强器是InstantiationModelAwarePointcutAdvisor,
                                                  判断每个增强器是否是AspectJPointcutAdvisor类型的,如果是就返回true
                                              2、否则永远返回false
                          2)if (bean != null) bean = applyBeanPostProcessorsAfterInitialization(bean, beanName);//这里即bean创建出来了需要进行一次后置处理
                    如果不能就往下走
                  2、doCreateBean();真正的去创建一个bean实例;和3.5的流程一样


 主配置类被拦截时会将切面类的属性的相关拦截器加入到AnnotationAwareAspectJAutoProxyCreator中
 1、MathCalculator:
     1、 MathCalculator调用postProcessBeforeInstantiation()方法时不会返回bean,即通过doCreateBean创建对象
     2、doCreateBean过程中:在applyBeanPostProcessorsAfterInitialization()方法中调用postProcessAfterInitialization()方法
          return wrapIfNecessary(bean, beanName, cacheKey);//即如果需要的情况下包装bean
          1、获取当前bean的所有增强器(通知方法) Object[] specificInterceptors
              1、找到候选的所有增强器(找哪些通知方法是需要切入当前bean方法)
              2、获取到能在当前bean使用的增强器
              3、给增强器排序
          2、保存当前bean在advisedBeans中
          3、如果当前bean需要增强,创建当前bean的代理对象
              1、获取所有增强器(通知方法)
              2、保存到proxyFactory中
              3、创建代理对象:Spring自动决定;最后创建出com.npu.aop.MathCalculator@39a8312f
                  JdkDynamicAopProxy(config);jdk动态代理;
                  ObjenesisCglibAopProxy(config);cglib动态代理
          4、给容器中返回当前组件使用cglib增强了的代理对象
          5、以后容器中获取到的就是这个组件的代理对象,执行目标方法的时候,代理对象就会执行通知方法的流程
      3、目标方法的执行
          容器中保存了组件的代理对象(cglib增强后的对象),这个对象里面保存了详细信息(比如增强器,目标对象,xxx)
          1、CglibAopProxy.intercept();拦截目标方法的执行
          2、根据ProxyFactory对象获取目标方法将要执行的拦截器链getInterceptorsAndDynamicInterceptionAdvice()
                  1、创建一个List<Object> interceptorList = new ArrayList<>(advisors.length);保存所有拦截器链
                     有一个默认的ExposeInvocationInterceptor.ADVISOR和自己配置的四个InstantiationModelAwarePointcutAdvisor
                  2、遍历所有增强器,将其转为MethodInterceptor
                      registry.getInterceptors(advisor):1、如果是 MethodInterceptor,直接加入到List中返回
                                                         2、如果不是,使用AdvisorAdapter将增强器转为MethodInterceptor
                                                                  三类适配器:1、MethodBeforeAdviceAdapter
                                                                             2、AfterReturningAdviceAdapter
                                                                             3、ThrowAdviceAdapter
                                                         3、转换完成,返回MethodInterceptor数组,即拦截器链
                                                              拦截器链:每一个通知方法又被包装成拦截器链,利用MethodInterceptor机制
          3、如果没有拦截器链,直接执行目标方法
          4、如果有拦截器链,把需要执行的目标对象,目标方法,拦截器链等信息传入创建一个CglibMethodInvocation对象,并调用proceed()方法获取返回值。
          5、拦截器链的触发过程(proceed()的方法中),拦截器链的机制保证通知方法与目标方法的执行顺序
                  1、如果没有拦截器,或者拦截器的索引和拦截器数组-1大小一样(执行到最后一个拦截器),直接执行目标方法后返回;
                      currentInterceptorIndex=this.interceptorsAndDynamicMethodMatchers.size() - 1
                  2、获取到当前下标拦截器interceptorOrInterceptionAdvice =this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex)(将当前拦截器的索引+1)
                  3、调用((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);
                      this为cglib的代理对象,再调用proceed,直到MethodBeforeAdviceInterceptor时会调用前置通知,然后再调用proceed。
                  4、MethodBeforeAdviceInterceptor方法返回后执行目标方法后到AspectJAfterAdvice的proceed中,执行后置通知
                  5、AspectJAfterAdvice执行完成后又返回到AfterReturningAdviceInterceptor方法中,
                      执行afterReturning()方法(方法执行过程中间不出任何异常,如果出了异常会被上层AspectJAfterThrowingAdvice捕获)
                  6、如果出现异常AspectJAfterThrowingAdvice会捕获到执行invokeAdviceMethod()方法
                  7、随后回到最初的proceed方法,返回,过程中一共进行了5次压栈操作、
 总结:
      1、@EnableAspectJAutoProxy 开启AOP功能
      2、@EnableAspectJAutoProxy会注册一个AnnotationAwareAspectJAutoProxyCreator组件
      3、AnnotationAwareAspectJAutoProxyCreator是一个后置处理器
      4、容器创建流程:
              1:registerBeanPostProcessor()注册所有后置处理器,会创建AnnotationAwareAspectJAutoProxyCreator对象
              2:finishBeanFactoryInitialize()初始化剩下的单实例bean
                  1)创建业务逻辑组件和切面组件
                  2)AnnotationAwareAspectJAutoProxyCreator会拦截组件的创建过程
                  3)组件创建完成之后,判断组件是否需要增强
                      是:切面的通知方法包装成增强器(Advisor);给业务逻辑组件创建一个代理对象(cglib代理)
      5、执行目标方法:
          1、代理对象执行目标方法
          2、CglibProxy.intercept()
              1、得到目标方法的拦截器链(增强器包装成拦截器MethodInterceptor)
              2、利用拦截器的链式机制,依次进入每一个拦截器进行执行;
              3、效果:
                  正常执行:前置通知->目标方法->后置通知->返回通知
                  异常执行:前置通知->目标方法->后置通知->异常通知

image.png

4. Spring 声明事务


  声明式事务:
 
  环境搭建:
       1、导入相关依赖
           数据源、数据库、SpringJdbc模块
       2、配置数据源、JdbcTemplate操作数据
       3、给方法上面标注@Transactional 标识当前方法是一个事务方法
       4、@EnableTransactionManagement开启基于注解的事务管理功能
       5、配置事务管理器来控制事务 public PlatformTransactionManager platformTransactionManager
  原理:
       1、@EnableTransactionManagement利用
                   TransactionManagementConfigurationSelector给容器中导入两个组件
                   1、AutoProxyRegistrar,它会给容器中注册一个InfrastructureAdvisorAutoProxyCreator组件
                                          InfrastructureAdvisorAutoProxyCreator也是一个后置处理器,利用后置处理器机制在对象创建以后包装对象,返回一个代理对象(增强器),
                                          代理对象执行方法利用拦截器链进行调用
                   2、ProxyTransactionManagementConfiguration
                           1、它会给容器中注册事务增强器、
                                   1、事务增强器要用事务注解的信息,AnnotationTransactionAttributeSource解析事务注解
                                   2、事务增强器需要事务拦截器:TransactionInterceptor保存了事务的属性信息,事务管理器
                                       TransactionInterceptor(它是一个MethodIntercepter)在目标方法执行的时候执行拦截器链
                                           事务拦截器:
                                               1、先获取事务相关的属性
                                               2、在获取PlatformTransactionManager,没有事先没有指定,最终会从容器中按照类型获取一个TransactionManager
                                               3、执行目标方法,如果异常,获取到事务管理器,利用事务管理器回滚操作,如果正常,利用事务管理器提交事务
 

4.1 扩展原理


 扩展原理
 BeanPostProcessor:bean后置处理器,bean创建对象初始化前后进行拦截工作的
 1、BeanFactoryPostProcessor:beanFactory的后置处理器,在beanFactory标注初始化后调用,所以bean的定义已经保存加载到beanFactory,但是bean的实例还未创建
      1、ioc容器创建对象
      2、执行invokeBeanFactoryPostProcessors(beanFactory);执行BeanFactoryPostProcessors
          如何找到所有的BeanFactoryPostProcessor并执行它们的方法:
              1、String[] postProcessorNames =beanFactory.getBeanNamesForType(BeanFactoryPostProcessor.class, true, false);
              2、在初始化创建其它组件前面执行

 2、BeanDefinitionRegistryPostProcessor是BeanFactoryPostProcessor的子接口,BeanDefinitionRegistry是Bean定义信息的保存中心,BeanFactory就是按照其中保存的bean的定义信息创建bean实例的
      postProcessBeanDefinitionRegistry()方法,在所有bean定义信息将要被加载到,但是bean实例还未创建,优先于BeanFactoryPostProcess执行,可以利用其给容器中再来添加一些组件
 原理:
      1)、ioc容器创建对象
      2)、执行执行invokeBeanFactoryPostProcessors(beanFactory);
      3)、从容器中获取到所有的BeanDefinitionRegistryPostProcessor组件beanFactory.getBeanNamesForType(BeanDefinitionRegistryPostProcessor.class, true, false);
          1、先触发 postProcessBeanDefinitionRegistry()方法
          2、再触发 postProcessBeanFactory()方法
      4)、再来从容器中执行BeanFactoryPostProcessor类型的组件,然后依次触发postProcessBeanFactory()方法

 3、ApplicationListener:监听容器中发布的事件,事件驱动模型的开发
      ApplicationListener<E extends ApplicationEvent>
      监听ApplicationEvent及其子类的相关事件
   步骤:
      1)、写一个监听器来监听某个事件(ApplicationEvent及其子类)
          @EventListener(class={})可以在普通的业务逻辑组件上的方法监听事件
          原理:使用EventListenerMethodProcessor处理器来解析方法上的@EventListener注解,它实现了EventListenerMethodProcessor接口
                  SmartInitializingSingleton接口的原理:单实例bean全部创建完成后
                  1)ioc容器创建,refresh();
                  2)finishBeanFactoryInitialization(beanFactory);初始化剩下的单实例bean;
                      1)一顿遍历先创建所有的单实例bean;
                      2)获取有创建好的单实例bean,判断是否是实现了 SmartInitializingSingleton接口类型的,
                          如果是就调用该接口的afterSingletonsInstantiated()方法
      2)、把监听器加入到容器中
      3)、只要容器中有相关类型的事件的发布,就能监听到这个事件
              ContextRefreshedEvent:容器刷新完成(所有bean都完全创建)会发布这个事件
              ContextClosedEvent:关闭容器发布这个事件
      4)、自定义发布一个事件 ioc容器.publishEvent(ApplicationEvent);

    原理:
       ContextRefreshedEvent、IOCTest_Ext$1、ContextClosedEvent
       1、ContextRefreshedEvent事件:
          1)容器创建对象:refresh();
          2)finishRefresh()方法中调用publishEvent(new ContextRefreshedEvent(this));
      2、自己发布的事件 publishEvent();
      3、ContextClosedEvent:close方法调用doClose方法发布ContextClosedEvent事件

       【事件发布流程】即publishEvent方法:
           1、获取事件的多播器:getApplicationEventMulticaster();
           2、调用multicastEvent(applicationEvent, eventType)派发事件
           3、获取到所有的ApplicationListener,即getApplicationListeners()
                1、如果有Executor,可以支持使用Executor进行异步派发
                2、否则同步的方式直接执行invokeListener(listener, event);
               拿到listener回调onApplicationEvent方法
        【事件的多播器【派发器】】
           1、容器创建对象:refresh()中
           2、initApplicationEventMulticaster();会初始化多播器
                  1、先去容器中有没有id="applicationEventMulticaster"的组件
                  2、如果没有,new SimpleApplicationEventMulticaster(beanFactory);同时注册到容器中,我们就可以在其它组件要派发事件,自动注入这个派发器
        【容器中有哪些监听器】
           1、容器创建对象:refresh()中
           2、registerListeners();
              从容器中拿到所有的监听器,把他们注册到applicationEventMulticaster中;
              String[] listenerBeanNames = getBeanNamesForType(ApplicationListener.class, true, false);
              //将listener注册到多播器中
              for (String listenerBeanName : listenerBeanNames)
                    getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);

5. Spring容器的创建过程

Spring容器的refresh()【创建刷新】

1、prepareRefresh()刷新前的预处理;
​       1)、initPropertySources();初始化一些属性设置;子类自定义个性化属性设置方法;
      ​ 2)、getEnvironment().validateRequiredProperties();检验属性的合法等
     ​  3)、this.earlyApplicationEvents = new LinkedHashSet<>();保存容器中的一些早期事件

2、obtainFreshBeanFactory();获取BeanFactory;
​     1)、refreshBeanFactory();刷新【创建】beanFactory 在GenericApplicationContext对象构造时this.beanFactory=new DefaultListableBeanFactory();设置id
     2)、getBeanFactory(); 返回刚才GenericApplicationContext创建的BeanFactory【DefaultListableBeanFactory】对象;
3、prepareBeanFactory(beanFactory);BeanFactory的预准备工作(BeanFactory进行一些设置)
    ​ 1)、设置BeanFactory的类加载器、支持表达式解析器
​     2)、添加部分BeanPostProcessor【ApplicationContextAwareProcessor】
    ​ 3)、设置忽略的自动装配的接口EnvironmentAware、EmbeddedValueResolverAware、xxx
​     4)、注册可以解析的自动装配;我们能直接在任何组件中自动注入:BeanFactory、ResourceLoader、ApplicationEventPublisher、ApplicationContext
​     5)、添加BeanPostProcessor【ApplicationListenerDetector】
​     6)、添加编译时的AspectJ;
​     7)、给BeanFactory中注册一些能用的组件:
            ​ environment【ConfigurableEnvironment】、 systemProperties【Map<String,Object>】、systemEnvironment【Map<String,Object>】
4、postProcessBeanFactory(beanFactory);BeanFactory准备工作完成后进行的后置处理工作;
    ​ 1)、子类通过重写这方法在BeanFactory创建并预准备完成以后做进一步的设置
              ===================以上是BeanFactory的创建及预准备工作=====================
5、invokeBeanFactoryPostProcessors(beanFactory);执行BeanFactoryPostProcessor的方法,BeanFactoryPostProcessor:BeanFactory的后置处理器,在BeanFactory标注初始化之后执行两个接口:BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessor接口
​     1)、执行BeanFactoryPostProcessor的方法:
            ​ 1)、获取所有BeanDefinitionRegistryPostProcessor
​             2)、先执行实现了PriorityOrdered优先级接口的、再执行实现了Ordered的接口的、最后执行其它的
            3)、获取所有BeanFactoryPostProcessor
​     2)、先执行实现了PriorityOrdered优先级接口的、再执行实现了Ordered的接口的、最后执行其它的
6、registerBeanPostProcessors(beanFactory);注册BeanPostProcessor,BeanPostProcessor(Bean的后置处理器)【拦截Bean的创建过程】不同类型的BeanPostProcessor,在Bean创建前后的执行时机是不一样的
​ 有如下几类: BeanPostProcessor、DestructionAwareBeanPostProcessor、​ InstantiationAwareBeanPostProcessor、SmartInstantiationAwareBeanPostProcessor、MergedBeanDefinitionPostProcessor【internalPostProcessors】
​      1)、获取所有的BeanPostProcessor; 后置处理器都默认可以通过PriorityOrdered、Ordered来指定优先级
      2)、先注册PriorityOrdered优先级接口的BeanPostProcessor把每一个BeanPostProcessor添加到BeanFactory中,beanFactory.addBeanPostProcessor(postProcessor);
    ​  3)、再注册了实现Ordered接口的
    ​  4)、最后注册其它的
    ​  5)、最终注册MergedBeanDefinitionPostProcessor类型的
​      6)、注册一个ApplicationListenerDetector;来在Bean创建完成后检查是否是ApplicationListener,addApplicationListener((ApplicationListener<?>) bean);
7、initMessageSource();初始化MessageSource组件(做国际化功能;消息绑定;消息解析等功能)
​   1)、获取BeanFactory
  ​ 2)、看容器中是否有id为messageSource,类型是MessageSource的组件如果有赋值给messageSource,如果没有自己创建一个DelegatingMessageSource;MessageSource:取出国际化配置文件中的某个key的值;能按照区域信息获取;
​   3)、把创建好的MessageSource注册在容器中,以后获取国际化配置文件的值的时候,可以自动注入MessageSource;调用其方法可以获得相关配置属性beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME, this.messageSource);

8、initApplicationEventMulticaster();初始化事件派发器【多播器】
​   1)、获取BeanFactory
​   2)、从BeanFactory获取applicationEventMulticaster的组件
​   3)、如果上一步没有配置;创建一个SimpleApplicationEventMulticaster
​   4)、将创建的ApplicationEventMulticaster添加到BeanFactory中,以后其他组件直接自动注入

9、onRefresh();留给子容器(子类)
  ​ 1)、子类重写这个方法,在容器刷新的时候可以自定义逻辑;

10、registerListeners();给容器中将所有项目里面的ApplicationListener注册进来
​   1)、从容器中拿到所有ApplicationListener组件
  ​ 2)、将每个监听器添加到事件派发器中getApplicationEventMulticaster().addApplicationListenerBean(listenerBeanName);
​   3)、派发之前步骤产生的事件;

11、finishBeanFactoryInitialization(beanFactory);初始化所有剩下的单实例bean
​   1)、beanFactory.preInstantiateSingletons();初始化剩下的单实例bean
       1)、获取容器中的所有Bean,依次进行初始化和创建对象List beanNames = new ArrayList<>(this.beanDefinitionNames);
  ​ 2)、遍历beanNames获取bean 的定义信息;
     RootBanDefinition bd = getMergedLocalBeanDefinition(beanName);
​   3)、Bean不是抽象的,是单实例的,是懒加载;
​       1)、判断是否是FactoryBean;是否是实现FactoryBean接口的Bean;
​       2)、如果是,利用工厂方法创建对象
      ​ 3)、不是工厂Bean,利用getBean(过程如下)(beanName);创建对象
​       4)、所有Bean都利用getBean创建完成以后;检查所有的Bean是否是SmartInitializingSingleton接口类型的,如果是就执行 afterSingletonsInstantiated()方法;

12、finishRefresh();完成BeanFactory初始化创建工作;IOC容器就创建完成
    ​ 1)、initLifecycleProcessor();初始化声明周期有关的后置处理器允许我们写一个LifecycleProcessor的实现类,可以在BeanFactory进行到特定生命周期时进行调用 默认从容器中找是否有LifeCycleProcessor的组件,如果没有,默认会创建一个 new DefaultLifecycleProcessor();然后加入到容器中
​     2)、getLifecycleProcessor().onRefresh();拿到所有前面定义的生命周期处理器回调onRefresh()方法
     3)、publishEvent(new ContextRefreshedEvent(this));发布容器刷新完成事件
     4)、LiveBeansView.registerApplicationContext(this);

6. SpringMVC与servlet3.0

1、web容器启动的时候,会扫描每个jar包下的META-INF/services/javax.servlet.ServletContainerInitializer
2、加载这个文件指定的SpringServletContainerInitializer
3、spring的应用一启动会加载感兴趣的WebApplicationInitializer接口下的所有组件
4、并且为WebApplicationInitializer组件创建对象(组件不是接口和抽象类)
​   1、AbstractContextLoaderInitializer;创建根容器createRootApplicationContext();
​   2、AbstractDispatcherServletInitializer:
​         创建一个web的ioc容器createServletApplicationContext();
​         创建一个DispatcherServlet:createDispatcherServlet(servletAppContext);
​         将创建的DispatcherServlet添加到ServletContext中:
      ​ servletContext.addServlet(servletName, dispatcherServlet);
​ 3、AbstractAnnotationConfigDispatcherServletInitializer:注解方式配置的DispatcherServlet初始化器
​ 创建根容器createRootApplicationContext();
​ getRootConfigClasses();传入一个配置类
​ 创建web的ioc容器:createServletApplicationContext();
​ getServletConfigClasses();获取配置类
总结:以注解方式来启动SpringMVC,继承AbstractAnnotationConfigDispatcherServletInitializer,实现抽象方法指定DispatcherServlet的配置信息
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342