SpringBootApplication
//修饰范围:类型
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
//修饰类子类继承
@Inherited
//注解1
@SpringBootConfiguration
//注解2
@EnableAutoConfiguration
//组件扫描
@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
//exclude给EnableAutoConfiguration注解exclude属性赋值
@AliasFor(annotation = EnableAutoConfiguration.class)
Class<?>[] exclude() default {};
//excludeName给EnableAutoConfiguration注解excludeName属性赋值
@AliasFor(annotation = EnableAutoConfiguration.class)
String[] excludeName() default {};
//scanBasePackages给ComponentScan注解basePackages属性赋值
@AliasFor(annotation = ComponentScan.class, attribute = "basePackages")
String[] scanBasePackages() default {};
//scanBasePackageClasses给ComponentScan注解basePackageClasses属性赋值
@AliasFor(annotation = ComponentScan.class, attribute = "basePackageClasses")
Class<?>[] scanBasePackageClasses() default {};
//nameGenerator给ComponentScan注解nameGenerator属性赋值
@AliasFor(annotation = ComponentScan.class, attribute = "nameGenerator")
Class<? extends BeanNameGenerator> nameGenerator() default BeanNameGenerator.class;
//proxyBeanMethods给Configuration注解proxyBeanMethods属性赋值
@AliasFor(annotation = Configuration.class)
boolean proxyBeanMethods() default true;
}
SpringBootApplication是一个组合注解,属性全部是EnableAutoConfiguration、Configuration、ComponentScan的别名,也就是通过这个注解给其他注解的属性赋值。
SpringBootConfiguration
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Configuration
public @interface SpringBootConfiguration {
//proxyBeanMethods给Configuration注解proxyBeanMethods属性赋值
@AliasFor(annotation = Configuration.class)
boolean proxyBeanMethods() default true;
}
相当于@Configuration
Configuration
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface Configuration {
//value给Component注解value属性赋值
@AliasFor(annotation = Component.class)
String value() default "";
boolean proxyBeanMethods() default true;
}
2:@EnableAutoConfiguration
源码如下
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
/**
* Exclude specific auto-configuration classes such that they will never be applied.
* @return the classes to exclude
*/
Class<?>[] exclude() default {};
/**
* Exclude specific auto-configuration class names such that they will never be
* applied.
* @return the class names to exclude
* @since 1.3.0
*/
String[] excludeName() default {};
}
重要注解有两个:
第一个注解@AutoConfigurationPackage
@Import(AutoConfigurationPackages.Registrar.class)
public @interface AutoConfigurationPackage {
}
@AutoConfigurationPackage该注解查看源码可知由@Import注解导入xxx类所以我们继续查看@Import注解的内容。
public @interface Import {
/**
* {@link Configuration @Configuration}, {@link ImportSelector},
* {@link ImportBeanDefinitionRegistrar}, or regular component classes to import.
*/
Class<?>[] value();
}
@Import注解常常用来导入组件,导入指定类型对象放入容器中。
所以@AutoConfigurationPackage这个注解的内容就是将AutoConfigurationPackages.Registrar.class这个类注入到容器中。
其中Registrar是AutoConfigurationPackages的一个内部类。
registerBeanDefinitions方法源码如下:
@Override
public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
register(registry, new PackageImport(metadata).getPackageName());
}
第二个参数 ==new PackageImport(metadata).getPackageName() ==是注解所在的包路径,例如:com.example.demo,该方法执行registry方法,将包路径下所有组件注入到容器中。
public static void register(BeanDefinitionRegistry registry, String... packageNames) {
if (registry.containsBeanDefinition(BEAN)) {
BeanDefinition beanDefinition = registry.getBeanDefinition(BEAN);
ConstructorArgumentValues constructorArguments = beanDefinition.getConstructorArgumentValues();
constructorArguments.addIndexedArgumentValue(0, addBasePackages(constructorArguments, packageNames));
}
else {
GenericBeanDefinition beanDefinition = new GenericBeanDefinition();
beanDefinition.setBeanClass(BasePackages.class);
beanDefinition.getConstructorArgumentValues().addIndexedArgumentValue(0, packageNames);
beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);
registry.registerBeanDefinition(BEAN, beanDefinition);
}
}
第二个注解@Import(AutoConfigurationImportSelector.class)
将 AutoConfigurationImportSelector 类加入容器中进行管理。
类中重要方法selectImports
@Override
public String[] selectImports(AnnotationMetadata annotationMetadata) {
if (!isEnabled(annotationMetadata)) {
return NO_IMPORTS;
}
AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader
.loadMetadata(this.beanClassLoader);
AutoConfigurationEntry autoConfigurationEntry = getAutoConfigurationEntry(autoConfigurationMetadata,
annotationMetadata);
return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
}
方法中调用getAutoConfigurationEntry代码如下
protected AutoConfigurationEntry getAutoConfigurationEntry(AutoConfigurationMetadata autoConfigurationMetadata,
AnnotationMetadata annotationMetadata) {
if (!isEnabled(annotationMetadata)) {
return EMPTY_ENTRY;
}
AnnotationAttributes attributes = getAttributes(annotationMetadata);
List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
configurations = removeDuplicates(configurations);
Set<String> exclusions = getExclusions(annotationMetadata, attributes);
checkExcludedClasses(configurations, exclusions);
configurations.removeAll(exclusions);
configurations = filter(configurations, autoConfigurationMetadata);
fireAutoConfigurationImportEvents(configurations, exclusions);
return new AutoConfigurationEntry(configurations, exclusions);
}
3:@ComponentScan
组件扫描注解