有时候我们会出现存在一个bean,但是找不到的情况,出现的原因大致有两个,一是未加注解,让spring进行管理,二是没有被spring扫描到,今天想讲一下第二个原因,spring没有扫描到bean。
要解这个问题,我们需要先知道spring是怎么知道需要扫描哪些地方的,对于一个springboot项目,关键点在于BootstrapApplication.java中使用了@SpringBootApplication注解。
而这个注解点进去可以看到它继承了很多注解。其中的关键就是@ComponentScan注解
注解类上也注明了当指定了basePackageClasses或者basePackages时候,将会扫描指定的包,但是当没有指定的时候,扫描的是声明了这个注解的类所在的包。
接下来我们看看这个注解是如何做的:全局搜索ComponentScan查看相关实现类,我们可以找到一个叫做ComponentScanAnnotationParser的类,通过类的命名我们可以知道这个就是@ComponentScan注解的解析器。里面有一个关键方法parse。我们先不用管各种属性,bean是怎么进来的,我们先看这样一段关键代码。这段代码首先读取了属性basePackages的值放入basePackages的set中,然后读取了basePackageClasses的值放入basePackages中,如果此时basePackages不为空,即我们指定了扫描包,则后续扫描的部分就是basePackages中的包,但是basePackages如果为空,此时取的是declaringClass的包,继续往外查看调用的话可以知道这个最终就是我们启动类,最终取的就是启动类所在的包。
所以,当我们的bean与启动类放在一个包下的时候,我们不用配置任何东西,也能扫描到,因为默认扫描的就是启动类所在的包。但是大多数情况下,我们都是做了分层处理,根据bean的不同作用放在不同的包下,比如controller,service等等,这时候我们就需要使用配置basePackages或者basePackageClasses。