问题点:在springboot 多模块项目中,新建了一个子模块,然后子模块的其中一个类注入了另一个模块的类,启动项目出现Could not autowire. No beans of 'SysLoginService' type found.的报错。
如上图所示,新建子模块youxi-mobile,有个WxLoginController控制器,类中利用@Autowired注解将
youxi-framework 中的SysLoginService类属性注入到里面。
结果就出现了这样的错误,无法自动注入,找不到SysLoginService类
首先我排查了youxi-mobile的pom.xml文件,看是否引入youxi-framework 的依赖,发现并没有问题
查找资料发现一种解决方法,就是在启动类的@SpringBootApplication注解上加上scanBasePackages = "com"
加上后发现没问题了
这时候我就有点奇怪,那为什么我的另一个子模块youxi-admin的启动类上并没有加上scanBasePackages = "com",同样跟上面一样进行属性注入就不会报错。
然后继续谷歌,又发现了一个知识点
有这样一个注解@ComponentScan,源码
package org.springframework.context.annotation;
@Retention(RetentionPolicy.RUNTIME)
//表示只可以声明在类上
@Target({ElementType.TYPE})
@Documented
@Repeatable(ComponentScans.class)
public @interface ComponentScan {
@AliasFor("basePackages")
String[] value() default {};
@AliasFor("value") //表示和value等价
String[] basePackages() default {};
Class<?>[] basePackageClasses() default {};
Class<? extends BeanNameGenerator> nameGenerator() default BeanNameGenerator.class;
@ComponentScan 的作用就是根据定义的扫描路径,把符合扫描规则的类装配到spring容器中
本质上,SpringBootApplicatioscan中的scanBasePackages属性底层原理正是复用了@ComponentScan,因此语法和意义基本一致
public @interface SpringBootApplication {
//复用了@ComponentScan
@AliasFor(
annotation = ComponentScan.class,
attribute = "basePackages"
)
String[] scanBasePackages() default {};
@AliasFor标签表示别名的意义,可以理解为等价于
因此,下面2种用法相同:
@SpringBootApplication (scanBasePackages="com.youxi")
public class YouXiApplication{
package com.test;
@ComponentScan("com.youxi")
@SpringBootApplication
public class YouXiApplication{
SpringBoot项目中
当使用Spring Boot项目时,可以不指定加载路径,即不使用@ComponentScan或@SpringBootApplication (scanBasePackages="com.youxi"),默认会加载YouXiApplication所在的包
举个例子,看下面定义的类:
package com.youxi;
@SpringBootApplication
public class YouXiApplication{
YouXiApplication的package为com.youxi,这个类使用了@SpringBootApplication注解,该注解定义了Spring将自动扫描包com.youxi及其子包下的bean。
如果你项目中所有的类都定义在com.youxi包及其子包下,那你不需要做任何事。
但假如你一个类定义在包com.youxi2下,则你需要将这个新包也纳入扫描的范围,有两个方案可以达到这个目的:
方案1, 定义@CoponentScan(“com”)
这么做扫描的范围扩大到整个父包com
方案2, 定义分别扫描两个包
@ComponentScan({“com.youxi”,”com.youxi2”})
所以经过我的观察,我发现我的youxi-admin项目的启动类是在com.youxi目录下
根据上面的说法,会自动扫描自动扫描包com.youxi及其子包下的bean。查看目录结构,youxi-framework项目中SysLoginService类所在的包确实是在com.youxi 目录下,所以当youxi-admin项目启动时,就会自动扫描到。
这时候看一下新创建的youxi-mobile
真相大白了!!YouxiMobileApplication启动类在com.youxi.mobile这个包目录下,所以项目启动后会到com.youxi.mobile这个目录下扫描有没有SysLoginService类,很显然,SysLoginService类并不存在于com.youxi.mobile下,所以这里只需要把启动类所在包目录改成com.youxi就可以了!
总结
虽然这只是个小问题,但也能反应出我对这方面的了解还是不够深不够扎实,所以写下了这篇文章,记录我踩过的坑,给大家提供一个解决这方面问题的思路同时方便以后查看~