The previous chapters were all concepts: review

  • BeanDefinition is the material
  • Bean is a finished product
  • BeanFactory is a warehouse that stores materials and finished products
  • ApplicationContext initializes the collection of materials into the repository, triggers the production line, and extracts the material production Bean

This article deals with the initialization of the ApplicationContext in the SpringBoot environment, mostly in the form of annotated beans.

Important component

Acting PostProcessorRegistrationDelegate: the post processors utility class

Postprocessors come in two types:

  • Spring BeanFactoryPostProcessor: occurrence in BeanDefinition collect storage stage
  • BeanPostProcessor: Occurs during the BeanDefintion generation Bean phase

BeanFactoryPostProcessor is subdivided into two categories:

  • BeanDefinitionRegistryPostProcessor: spring BeanFactoryPostProcessor interface inheritance. Its postProcessBeanDefinitionRegistry interface method, has registered more Bean semantics.
  • BeanFactoryPostProcessor: Semantics biased toward modification.

ConfigurationClass: this class represents a ConfigurationClass that encapsulates a BeanDefinition with configuration semantics. So how do we understand this. I think of it as an XML file. Remember when you first started using Spring, you needed to configure beans in XML.

ConfigurationClass semantics:

  • A ConfigurationClass can be thought of as an XML file. Note: @Configuration
  • A ConfigurationClass can define beans. Similar to an XML, you can define Bean information. Represents the annotation @bean
  • A ConfigurationClass can introduce other ConfigurationClasses, just as XML files introduce other XML files. Representative notes: @import,@ComponentScan, etc

So, how do we know that it’s a ConfigurationClass? Look at the following

3. ConfigurationClassUtils: ConfigurationClass tools

CheckConfigurationClassCandidate () method: first the BeanDefinition annotation information.

  • ConfigurationClass full ConfigurationClass. It is identified by the @Configuration annotation. If so, it must be ConfigurationClass
  • Determine whether isLiteConfigurationCandidate, judgment will be @ Component, @ ComponentScan, @ Import, @ ImportResource annotations. Or a method with the @bean identifier

Basically, it depends on whether it can output beans. Determines whether it belongs to a ConfigurationClass

4. ConfigurationClassPostProcessor: belongs to a kind of post processor. Implements the BeanDefinitionRegistryPostProcessor, indirect spring BeanFactoryPostProcessor is realized. So it has the dual functions of registration and modification. The main job is to handle ConfigurationClass

5. ConfigurationClassParser: ConfigurationClass parser. Be able to parse the ConfigurationClass in your project (by analogy, find all the XML configured with beans in your project)

6. ConfigurationClassBeanDefinitionReader: BeanDefintion reader, read it from ConfigurationClass Bean definitions (analogy can be read out from the XML beans)

Collection and storage:

It is recommended to read with source code

BeanDefintion gathered in the refresh () initialization method of invokeBeanFactoryPostProcessors (the beanFactory) stage, spring BeanFactoryPostProcessor execution.

ApplicationContext entrust PostProcessorRegistrationDelegate tools perform post Processors. PostProcessorRegistrationDelegate. InvokeBeanFactoryPostProcessors performs all spring BeanFactoryPostProcessor () method

InvokeBeanFactoryPostProcessors: can be divided into two steps

  • First BeanDefinitionRegistryPostProcessor execution. To call its postProcessBeanDefinitionRegistry BeanDefinition registration
  • BeanFactoryPostProcessor then calls postProcessBeanFactory to modify the BeanDefinition function.

ConfigurationClassPostProcessor as a BeanDefinitionRegistryPostProcessor. Is carried out in the first place. Its processConfigBeanDefinitions methods: where is the beginning of the collection BeanDefinition.

ConfigurationClassPostProcessor collection steps:

  1. ConfigurationClass can be found in the repository. Since it is early, the only ConfigurationClass in the repository is the springBoot startup file. This ConfigurationClass can also be thought of as a top-level ConfigurationClass, just as a top-level XML file is unassociated with other XML files. The startup file ConfigurationClass exists as an entry point.
  2. Create a ConfigurationClassParser to parse the ConfigurationClass.
  3. call**ConfigurationClassParser.parse()**Parse the ConfigurationClass you just found. Through this entry. Find all required ConfigurationClasses. It’s kind of like finding all the XML associated with an XML file. Understanding by analogy.
  4. To create a**ConfigurationClassBeanDefinitionReader.loadBeanDefinitions() **From all of the ConfigurationClasses found. Iterating reads the Bean definitions in each ConfigurationClass.

Branch resolution:

The third step ConfigurationClassParser. Parse () branchConfigurationClass: Object to find all required ConfigurationClass.

This branch is roughly divided into two parts:

  • The parse () : processConfigurationClass () – > doProcessConfigurationClass () :
  • ProcessDeferredImportSelectors () : ImportSelector treatment need to delay. Lazy loading here will eventually follow the processImports() logic.

Since the second step is ultimately the first step, only the first step is studied.

DoProcessConfigurationClass: ConfigurationClass: (parse process involves a lot of recursion. The current ConfigurationClass is stored via an ArrayDeque dual-ended stack.)

  1. ProcessMemberClasses recursively processes inner classes. Check if the inner class of the current ConfigurationClas is a ConfigurationClass. If it is a recursive processConfigurationClass () – > doProcessConfigurationClass ().
  2. Handle @propertysource annotations
  3. Handling @ComponentScan annotation: There is also a recursive operation here. The scanned set of definitions is checked for any further configuration classes and recursively resolved as needed. This concentration are defined with the ComponentScanAnnotationParser parser to find all the @ Component annotation of the class. Think of it as a ConfigurationClass, too. Then recursively parse the ConfigurationClass
  4. ProcessImports () recursively resolves the configuration classes introduced by the @import annotation. The @import imported classes are handled in three different ways.
  5. Handle the @importResource annotation. Add the imported resource value to the importedResources property of the current ConfigurationClass.
  6. Handling @Beans: Check to see if the current ConfigurationClass has methods for @Beans. Add it to the beanMethods property of the ConfigurationClass.
  7. If the superclass exists and is notjavaClass in the package and has not yet been processed to return it for the outer loop to continue. This is also why doProcessConfigurationClass would be nested in a do while loop.

ProcessImports (): @import The imported classes are handled in three cases.

Import a class that implements the ImportSelector interface.

  • If DeferredImportSelector is implemented, it is treated as deferred. In deferredImportSelectors cache in processDeferredImportSelectors () branch
  • If you implement ImportSelector, perform the selectImports method recursively to handle a ConfigurationClass that might be annotated by @import

(2) the imported is an implementation of a ImportBeanDefinitionRegistrar interface class:

  • Indicates the need to manually register beans for operations in the container. The implementation class added to the current ConfigurationClass importBeanDefinitionRegistrars attributes.

(3) The normal class is treated as a ConfigurationClass for recursive parsing.

  • Think of it as a ConfigurationClass recursively.

Summary: After some series recursion, analysis. The end result is a ConfigurationClass. ConfigurationClass carries various Bean definitions.

This is also where we can see something about registering components in a container

  • Annotations @ Controller / @ Service / @ the Repository / @ Component
  • The Bean returned by @bean.
  • @import Quickly imports a component into a container: a normal class or ImportSelector interface implementation class; Or ImportBeanDefinitionRegistrar implementation class
  • ImportResource Imports an XML file.

ConfigurationClassBeanDefinitionReader.loadBeanDefinitions()Branch: The purpose is to parse the BeanDefinition ConfigurationClass.

So with ConfigurationClass, here’s the parsing. Just like having an XML file, you can parse the Bean definition in the XML. This section explains how beanDefinitions in ConfigurationClass are resolved.

Loop through all of the found ConfigurationClasses

  • Configclass.isimported () : Is the current ConfigurationClass imported by someone else via @import? Yes, it is injected into the repository as a BeanDefinition.
  • ConfigClass. GetBeanMethods () : @ Bean annotations Peugeot method return values parsed into a BeanDefinition injected into the warehouse.
  • LoadBeanDefinitionsFromImportedResources (configClass getImportedResources ()) : Loads the Bean definition in the XML file introduced by @importResource in the current ConfigurationClass. Specifically, the XmlBeanDefinitionReader reader is used to read Bean definitions configured in the XML, which are parsed into BeanDefinitions and injected into the repository.
  • LoadBeanDefinitionsFromRegistrars (configClass getImportBeanDefinitionRegistrars ()) : Remove implements ImportBeanDefinitionRegistrar interface classes, perform its registerBeanDefinitions method. Register the appropriate BeanDefinition into the repository.

Summary:

The collection of BeanDefinitions is completed by parsing the BeanDefinitions from the ConfigurationClass and putting them into the repository.

The class ConfigurationClass itself is also a BeanDefinition if the @Controller/ @service /@Repository/@Component Component is not the annotation.

LoadBeanDefinitions () parses registered Bean methods for the last three:

  • @ Bean annotation
  • @import three introduced.
  • @importResource Imported XML file.

Conclusion:

To sum up, the collection and warehousing stage of BeanDefinition is actually a process of finding the ConfigurationClass of the Bean definition, parsing the BeaDefinition in the file, and then storing the BeaDefinition in the repository.

Welcome everyone to pay attention to my public number [source code action], the latest personal understanding timely.