spring boot启动不扫描创建bean怎么回事

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring boot启动不扫描创建bean怎么回事相关的知识,希望对你有一定的参考价值。

说明你的spring boot启动时的application类不在io.github.gefangshuai.app及其子包下。

SpringBoot项目的Bean装配默认规则是根据Application类所在的包位置从上往下扫描的。“Application类”是指SpringBoot项目入口类。如果Application类所在的包为:io.github.gefangshuai.app,则只会扫描io.github.gefangshuai.app包及其所有子包,如果service或dao所在包不在io.github.gefangshuai.app及其子包下,则不会被扫描。

改变这种扫描包的方式的原理很简单:用@ComponentScan注解进行指定要扫描的包以及要扫描的类。

可以用以下方式测试:

第一步:新建两个包cn.kfit ; org.kfit;

第二步:新建两个测试类;

在这里为了方便测试,我们让我们的类在启动的时候就进行执行,所以就编写两个类,实现接口CommandLineRunner,这样在启动的时候我们就可以看到打印信息了。

cn.kfit.MyCommandLineRunner1  : 

package cn.kfit;  
  
import org.springframework.boot.CommandLineRunner;  
  
@Configuration  
publicclass MyCommandLineRunner1 implements CommandLineRunner   
  
    @Override  
    publicvoid run(String... args) throws Exception   
       System.out.println("MyCommandLineRunner1.run()");  
  
      

org.kfit.MyCommandLineRunner2  : 

package org.kfit;  
  
import org.springframework.boot.CommandLineRunner;  
  
  
@Configuration  
publicclass MyCommandLineRunner2 implements CommandLineRunner   
  
    @Override  
    publicvoid run(String... args) throws Exception   
  
       System.out.println("MyCommandLineRunner2.run()");  
  
      
  

第三步:启动类进行注解指定;
在App.java类中加入如下注解:

//可以使用:basePackageClasses=,basePackages=  
@ComponentScan(basePackages="cn.kfit","org.kfit")

启动时如果看到打印信息:

则说明配置成功。

参考技术A 这几天一直被这个问题困扰,Spring boot启动没报错,就是扫描不到需要注册的类。今天终于发现,Spring boot的启动类需要放在需要扫描类的上层目录,这样才能扫描到同级或者子类需要注册的类。 但是在网上我发现一个其他问题,有人说用 @ComponentScan直接写需要扫描类的路径,但是我试验一直没成功。不知道为什么,麻烦帮忙看下! 刚接触spring boot 先谢谢各位了 。

下图是我要扫描的dao和service,但是启动后一直报错。报的错一看就是没扫描到类ReadingListRepository,这个类继承JpaRepository

2017-04-25 22:45:57.349 WARN 7864 --- [ main] ationConfigEmbeddedWebApplicationContext : Exception encountered during context initialization - cancelling refresh attempt: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'readingListController': Unsatisfied dependency expressed through field 'readingListRepository'; nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'org.dawn.oket.dao.ReadingListRepository' available: expected at least 1 bean which qualifies as autowire candidate. Dependency annotations: @org.springframework.beans.factory.annotation.Autowired(required=true)
2017-04-25 22:45:57.349 INFO 7864 --- [ main] j.LocalContainerEntityManagerFactoryBean : Closing JPA EntityManagerFactory for persistence unit 'default'
2017-04-25 22:45:57.349 INFO 7864 --- [ main] org.hibernate.tool.hbm2ddl.SchemaExport : HHH000227: Running hbm2ddl schema export
2017-04-25 22:45:57.349 INFO 7864 --- [ main] org.hibernate.tool.hbm2ddl.SchemaExport : HHH000230: Schema export complete
2017-04-25 22:45:57.359 INFO 7864 --- [ main] o.apache.catalina.core.StandardService : Stopping service Tomcat
2017-04-25 22:45:57.399 INFO 7864 --- [ main] utoConfigurationReportLoggingInitializer :
Error starting ApplicationContext. To display the auto-configuration report re-run your application with 'debug' enabled.
2017-04-25 22:45:57.681 ERROR 7864 --- [ main] o.s.b.d.LoggingFailureAnalysisReporter :
***************************
APPLICATION FAILED TO START
***************************
Description:
Field readingListRepository in org.dawn.oket.service.ReadingListController required a bean of type 'org.dawn.oket.dao.ReadingListRepository' that could not be found.
Action:
Consider defining a bean of type 'org.dawn.oket.dao.ReadingListRepository' in your configuration.
Disconnected from the target VM, address: '127.0.0.1:52934', transport: 'socket'
Process finished with exit code 1本回答被提问者采纳
参考技术B 纯手写答案:
我也遇到了这个问题,接下来是解决的方法
第一步:确保application在项目的根目录,这样就可以解决,如果不行,可以使用@ComponentScan或者@ComponentScans,@ComponScans的使用方法可查看源码。
如果第一步不行,
第二步:我遇到的情况就是第二种,使用idea创建springboot项目的时候没有勾选mybatis,导致生成的pom文件中没有写入mybatis jar包,然后我到maven仓库随便找了个mybatis的jar包直接导入了,问题出现,由于此mybatis并不是和springboot集成的jar包,导致出问题,
解决的方法:用<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>1.3.0</version>
</dependency>
取代pom文件中的artifactId 为mybatis的dependency
=。=
参考技术C 查看有没有其它的项目中 Application 没有在最上面的项目 前面的项目也可以 引起 后面的项目出现在这个问题

spring boot启动流程

参考技术A 入口是一个main方法,这个main方法里面new一个SpringApplication对象,传入bean源,就是注解了SpringBootApplication 的那个类,调用run方法。
spring boot启动流程分为两部分 一部分是准备阶段,一部分是运行阶段
准备阶段主要有这么几步:
1、配置bean的源,就是bean的来源,就是注解了SpringBootApplication的那个类。
2、推断,推断应用类型,有webflux,webservlet,none
推断主类,这个是通过线程堆栈实现,构造一个运行时异常,找异常堆栈里面找mian所在的那个类。
3、加载上下文初始化器 在这里可以调整applicationcontext
4、加载应用事件监听器 这两个东西都是在mete-info/spring.factories文件里面。

运行阶段,就是调用run方法
先是加载SpringApplication运行时监听器,SpringApplicationRunListeners,默认只有一个EventPublishingRunListener,需要传入准备阶段加载的listeners,在这个地方调用一个广播器广播事件。
然后会根据推断类型,创建一个ApplicationContext,ConfigurableApplicationContext
对这个context初始化,refresh方法
最后会返回这个context。

自动装配:
1、激活自动配置, EnableAutoConfiguration
2、实现自动配置(注解装配,模式装配,条件装配)配置bean
3、将这个配置配置到spring.factories文件里面去

以上是关于spring boot启动不扫描创建bean怎么回事的主要内容,如果未能解决你的问题,请参考以下文章

Spring Boot 之 spring.factories

Spring Boot是如何扫描和加载bean的

spring boot启动找不到ServletWebServerFactory bean

Spring Boot 注解大全

使用包含 bean 的外部 JAR 启动 Spring Boot 应用程序时出错

由于缺少 ServletWebServerFactory bean,Spring Boot jar 无法启动 Web 服务器