@AutoConfigurationPackage自动包规则原理

@AutoConfigurationPackage
AutoConfigurationPackage翻译过来就是自动配置包
路径:

@SpringBootApplication  -> @EnableAutoConfiguration -> @AutoConfigurationPackage
1
AutoConfigurationPackage.java

package org.springframework.boot.autoconfigure;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import org.springframework.context.annotation.Import;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@Import(AutoConfigurationPackages.Registrar.class)
public @interface AutoConfigurationPackage {

    String[] basePackages() default {};

    Class<?>[] basePackageClasses() default {};

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
我们发现@AutoConfigurationPackage其实它是一个@Import。@import是什么?@import就是给容器中导入一个组件,这个组件叫什么呢?叫AutoConfigurationPackages.Registrar的组件,但是这个Registrar作用是什么?

其实@import有更高级的用法,大家可以来参照:

@import高级用法

这里直接说意义是什么:我现在给容器中导的是一个Registrar

Registrar.java

    /**
     * {@link ImportBeanDefinitionRegistrar} to store the base package from the importing
     * configuration.
     */
static class Registrar implements ImportBeanDefinitionRegistrar, DeterminableImports {
        @Override
        public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
            register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0]));
        }

        @Override
        public Set<Object> determineImports(AnnotationMetadata metadata) {
            return Collections.singleton(new PackageImports(metadata));
        }

    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
我们发现它有俩个方法,其实也就是说我们利用Registrar给容器里面批量注册我们的这个组件(register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0]));),因为@import一个一个导太麻烦了 , 我们可以写一段代码批量注册:我们看到调用register方法

在这批量注册((registry, new PackageImports);我们批量注册什么呢?我们断点打到register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0]));

我们Debug一下。

所以我们看到我们的第一个核心@AutoConfigurationPackage它是利用AutoConfigurationPackages.Registrar给容器中导入组件但是这不是导入一个组件,是利用Registrar给容器中导入一系列的组件。那一系列组件?看源码

我们来看registerBeanDefinitions这个方法传了俩个参数:

一个是AnnotationMetadata(相当于我们注解的源信息)我们的注解是谁?注解指的就是@AutoConfigurationPackage这个注解;注解的源信息就代表了我们这个注解标在了哪、它的每一个属性值都是什么。我们打开源信息:


我们这个注解是标在哪的?这个注解就是标在主程序里面的,因为我们这个是合成注解@EnableAutoConfiguration包含它所以@AutoConfigurationPackage也是标注在它(主程序)里面的

所以我们看到注解的源信息里面我们标在哪个类上?introspectedClass={Class@1337}"class com.chentianyu.boot.MianApplication"(主程序)上

然后它在这里干了些什么事?它new了一个PackageImports, 相当于我们的整个包要导入的东西,它把这个元注解(metadata)拿到,有一个getPackageNames,相当于把我们注解源信息(metadata)拿来获取到我们这个包名(getPackageNames),我们怎么看包名(让它来计算一下)?

选择new PackageImports(metadata).getPackageNames()  ->  右键选择"Evaluate Expression"
1
就是com.chentianyu,boot。为什么得到的是包名?因为我们这个AutoConfigurationPackage注解标注在主类上,主类所在的包就是这个com.chentianyu,boot包,先得到了主类的包名然后把这个包名最终封装(toArray(new String[0]))到我们的一个数组里面,然后给我们注册(register())进去,相当于我们这个Registrar就是把某一个包下的所有组件批量注册进容器

@Import(AutoConfigurationPackages.Registrar.class)
public @interface AutoConfigurationPackage {}
//利用Registrar给容器中导入一系列组件
//将指定的一个包下的所有组件导入进来。哪个包下?主类所在的包下
1
2
3
4
这就解释了一个事情:为什么我们默认的包路径是主类所在的这个包
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值