文章目录
- 前言
- 一、ComponentScan 扫描路径定义:
- 二、包路径扫描:
- 2.1 AnnotationConfigApplicationContext:
- 2.1.1 reader 和 scan 初始化:
- 2.2 包扫描:
- 2.2.1 scan 扫描:
- 2.2.2 doScan 扫描:
- 2.2.3 findCandidateComponents 获取BeanDefinition :
- 三、扩展:
- 3.1 条件Condition :
- 3.2 Lookup 注解 :
- 总结
前言
Spring 中通过不同的bean 负责不同的功能,那么spring 在加载的时候,怎么知道哪些类需要去生成bean?
一、ComponentScan 扫描路径定义:
在spring 的启动类中可以添加 @ComponentScan 并且 定义扫描的bean路径,从而在spring 启动的时候,可以知晓从哪些地方的类需要去生成bean 。
@ComponentScan(basePackages = "com.xxxx")
main
public static void main(String[] args) {
ConfigurableApplicationContext context = new AnnotationConfigApplicationContext(MyConfig.class);
context.getBean("xxxx");
}
二、包路径扫描:
2.1 AnnotationConfigApplicationContext:
public AnnotationConfigApplicationContext(Class<?>... componentClasses) {
this(); // reader 和 scan 初始化
this.register(componentClasses);
this.refresh();// bean 的生成
}
2.1.1 reader 和 scan 初始化:
public AnnotationConfigApplicationContext() {
StartupStep createAnnotatedBeanDefReader = this.getApplicationStartup().start("spring.context.annotated-bean-reader.create");
// 定义BeanDefinitionReader
this.reader = new AnnotatedBeanDefinitionReader(this);
createAnnotatedBeanDefReader.end();
// 定义BeanDefinitionScanner 这里会把 Component.class 注解类默认放入到 includeFilters的 TypeFilter中
this.scanner = new ClassPathBeanDefinitionScanner(this);
}
2.2 包扫描:
通过 ClassPathBeanDefinitionScanner scan 方法,为每个要生成bean 的类 生成BeanDefinition,后面在创建bean 的时候可以从中获取类的注解,属性 ,方法等。
2.2.1 scan 扫描:
public int scan(String... basePackages) {
int beanCountAtScanStart = this.registry.getBeanDefinitionCount();
this.doScan(basePackages);// 做扫描工作
if (this.includeAnnotationConfig) {
AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
}
// 返回本次扫描了多少个BeanDefinition
return this.registry.getBeanDefinitionCount() - beanCountAtScanStart;
}
2.2.2 doScan 扫描:
protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
Assert.notEmpty(basePackages, "At least one base package must be specified");
Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet();
String[] var3 = basePackages;
int var4 = basePackages.length;
// 遍历每个包路径
for(int var5 = 0; var5 < var4; ++var5) {
String basePackage = var3[var5];
// 遍历获取每个路径下的 BeanDefinition
Set<BeanDefinition> candidates = this.findCandidateComponents(basePackage);
Iterator var8 = candidates.iterator();
while(var8.hasNext()) {
BeanDefinition candidate = (BeanDefinition)var8.next();
// 获取 这个类的 scope
ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
candidate.setScope(scopeMetadata.getScopeName());
// 获取 这个类的 bean 名称 如果 @Component 中定义了value 值则获取改制 否则按照默认的规则生成bean name
// 即 类的第一个字母小写.但是如果 类的名字第一个和
// 第二个字母都是大写,则直接使用,生成的bean 的首字母和第二个字母都是大写
String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);
if (candidate instanceof AbstractBeanDefinition) {
// BeanDefinition 赋值默认值(是否懒加载等)
this.postProcessBeanDefinition((AbstractBeanDefinition)candidate, beanName);
}
if (candidate instanceof AnnotatedBeanDefinition) {
// 处理 @Lazy @Primary,@DependsOn @Role @Description等注解
AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition)candidate);
}
// 如果改bean 的 BeanDefinition 已经存在大部分会抛异常出来
if (this.checkCandidate(beanName, candidate)) {
// 检查spring 中是否已经有改beanName 如果没有存在则进行注册
BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
beanDefinitions.add(definitionHolder);
// 注册最终放入到 DefaultListableBeanFactory 的 Map<String, BeanDefinition> beanDefinitionMap; 中
this.registerBeanDefinition(definitionHolder, this.registry);
}
}
}
return beanDefinitions;
}
2.2.3 findCandidateComponents 获取BeanDefinition :
public Set<BeanDefinition> findCandidateComponents(String basePackage) {
return this.componentsIndex != null && this.indexSupportsIncludeFilters() ? this.addCandidateComponentsFromIndex(this.componentsIndex, basePackage) : this.scanCandidateComponents(basePackage);
}
scanCandidateComponents: 扫描 @Component
private Set<BeanDefinition> scanCandidateComponents(String basePackage) {
// BeanDefinition set 集合将扫描到有 @Component 注解标注并且符合条件的包装成BeanDefinition
Set<BeanDefinition> candidates = new LinkedHashSet();
try {
// 定义class 的资源路径
String packageSearchPath = "classpath*:" + this.resolveBasePackage(basePackage) + '/' + this.resourcePattern;
// 获取包路径下的 file 资源
Resource[] resources = this.getResourcePatternResolver().getResources(packageSearchPath);
boolean traceEnabled = this.logger.isTraceEnabled();
boolean debugEnabled = this.logger.isDebugEnabled();
Resource[] var7 = resources;
int var8 = resources.length;
for(int var9 = 0; var9 < var8; ++var9) {
// 获取到每个file 资源
Resource resource = var7[var9];
if (traceEnabled) {
this.logger.trace("Scanning " + resource);
}
try {
// 获取改资源的 类元数据信息 (注解,属性,方法等)
MetadataReader metadataReader = this.getMetadataReaderFactory().getMetadataReader(resource);
if (this.isCandidateComponent(metadataReader)) {
// 符合生成bean 的条件,构造BeanDefinition
ScannedGenericBeanDefinition sbd = new ScannedGenericBeanDefinition(metadataReader);
sbd.setSource(resource);
if (this.isCandidateComponent((AnnotatedBeanDefinition)sbd)) {
if (debugEnabled) {
this.logger.debug("Identified candidate component class: " + resource);
}
// 加入 bean定义的set 集合
candidates.add(sbd);
} else if (debugEnabled) {
this.logger.debug("Ignored because not a concrete top-level class: " + resource);
}
} else if (traceEnabled) {
this.logger.trace("Ignored because not matching any filter: " + resource);
}
} catch (FileNotFoundException var13) {
if (traceEnabled) {
this.logger.trace("Ignored non-readable " + resource + ": " + var13.getMessage());
}
} catch (Throwable var14) {
throw new BeanDefinitionStoreException("Failed to read candidate component class: " + resource, var14);
}
}
return candidates;
} catch (IOException var15) {
throw new BeanDefinitionStoreException("I/O failure during classpath scanning", var15);
}
}
isCandidateComponent 是否符合bean 的条件: 这个类不是内部类,也不是接口 ,或者改类是抽象类但是没有定义 Lookup 则可以满足生成bean 的条件
protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException {
// 如果 类中定义 了排除 ,则进行匹配,如果该类在排除的范围之内则直接返回false
Iterator var2 = this.excludeFilters.iterator();
TypeFilter tf;
do {
if (!var2.hasNext()) {
// 如果没有被排除则,查看是否在 包含 filter 之内,如果都不包含也返回false
// 在容器准备阶段已经定义了一个 默认的 的includeFilter ,它就是 @Component
var2 = this.includeFilters.iterator();
do {
if (!var2.hasNext()) {
return false;
}
tf = (TypeFilter)var2.next();
} while(!tf.match(metadataReader, this.getMetadataReaderFactory()));
// 类中有@Component 注解,在进行条件匹配
return this.isConditionMatch(metadataReader);
}
tf = (TypeFilter)var2.next();
} while(!tf.match(metadataReader, this.getMetadataReaderFactory()));
return false;
}
是否条件匹配 isConditionMatch:得到true 则意味这个类需要生成baen, 得到false 则不需要生成bean
this.isConditionMatch(metadataReader);
private boolean isConditionMatch(MetadataReader metadataReader) {
if (this.conditionEvaluator == null) {
this.conditionEvaluator = new ConditionEvaluator(this.getRegistry(), this.environment, this.resourcePatternResolver);
}
return !this.conditionEvaluator.shouldSkip(metadataReader.getAnnotationMetadata());
}
public boolean shouldSkip(@Nullable AnnotatedTypeMetadata metadata, @Nullable ConfigurationCondition.ConfigurationPhase phase) {
if (metadata != null && metadata.isAnnotated(Conditional.class.getName())) {
if (phase == null) {
return metadata instanceof AnnotationMetadata && ConfigurationClassUtils.isConfigurationCandidate((AnnotationMetadata)metadata) ? this.shouldSkip(metadata, ConfigurationPhase.PARSE_CONFIGURATION) : this.shouldSkip(metadata, ConfigurationPhase.REGISTER_BEAN);
} else {
// 调用业务本身,改类 实现了Condition 接口 的matches 方法,如果得到true 则意味这个类需要生成baen
// 得到false 则不需要生成bean
List<Condition> conditions = new ArrayList();
Iterator var4 = this.getConditionClasses(metadata).iterator();
while(var4.hasNext()) {
String[] conditionClasses = (String[])var4.next();
String[] var6 = conditionClasses;
int var7 = conditionClasses.length;
for(int var8 = 0; var8 < var7; ++var8) {
String conditionClass = var6[var8];
Condition condition = this.getCondition(conditionClass, this.context.getClassLoader());
conditions.add(condition);
}
}
AnnotationAwareOrderComparator.sort(conditions);
var4 = conditions.iterator();
Condition condition;
ConfigurationCondition.ConfigurationPhase requiredPhase;
do {
do {
if (!var4.hasNext()) {
return false;
}
condition = (Condition)var4.next();
requiredPhase = null;
if (condition instanceof ConfigurationCondition) {
requiredPhase = ((ConfigurationCondition)condition).getConfigurationPhase();
}
} while(requiredPhase != null && requiredPhase != phase);
} while(condition.matches(this.context, metadata));
return true;
}
} else {
// 类上没有条件注解则直接 返回false ,意味着 这个类需要生成bean
return false;
}
}
三、扩展:
3.1 条件Condition :
Spring中的Condition接口是用于在runtime时决定一个bean或者某配置是否需要被创建或加载的。你可以使用@Conditional注解,并实现Condition接口,然后通过Condition#matches()方法来定义条件。
例如:
public class OnSystemPropertyCondition implements Condition {
@Override
public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
return.getProperty("my.property") != null;
}
}
在此例中,如果系统属性"my.property"存在,那么条件就满足,否则条件不满足。
然后你可以使用@Conditional注解来利用这个Condition:
@Configuration
@Conditional(OnSystemPropertyCondition.class)
public class ConditionConfig {
// some beans definitions...
}
在上面的例子中,只有当"my.property"系统属性存在时,ConditionConfig的定义和它下面的Bean才会被创建和注册到Spring的ApplicationContext中。
Condition提供了很好的方式来控制Bean的创建和配置的加载,使其可以适应不同的环境和条件。
3.2 Lookup 注解 :
Spring的@Lookup注解是用来覆盖在Spring IoC容器中单例Bean的某些方法,使它们每次被调用时都返回一个新的原型Bean(prototype bean)实例。
在某些情况下,可能需要在一个单例Bean中维护一些原型Bean的引用。但是,因为单例Bean在器启动时只被创建一次,它所持有的原型Bean也只会在那时创建一次。这可能会导致问题,因为通常我们期望每次请求原型Bean时,容器都能提供一个新的实例。
这时,我们就可以使用@Lookup注解来解决这个问题。@Lookup注解让我们可以定义一个返回特定类型的方法,并在需要时覆盖这个方法,使得每次调用这个方法时容器都会返回一个新的Bean实例。
@Service
public abstract class SingletonService {
// ...
@Lookup
public abstract PrototypeBean createPrototypeBean();
public void doSomething() {
PrototypeBean prototypeBean = createPrototypeBean();
// 处理prototypeBean
}
}
在上面的例子中,SingletonService是一个单例Bean,其中定义了一个用@Lookup注解标注的方法createPrototypeBean。每次调用这个方法,Spring都会为我们返回一个新的PrototypeBean实例。
总结
spring 通过sanner 对其传入的包路径进行扫描,遍历每个路径,获取其路径下的所以file 资源,筛选出有@Component 标注 并且非接接口 非抽象类,没有条件注解(或者条件注解返回true)的类,为其设置name ,填充socpe,lazy,Primary,DependsOn,Description 等属性包装生成BeanDefinition 最终放入到beanDefinitionMap ,供后面生成bean使用。