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类中加入如下注解:
@ComponentScan(basePackages="cn.kfit","org.kfit")
启动时如果看到打印信息:
则说明配置成功。
下图是我要扫描的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启动找不到ServletWebServerFactory bean
使用包含 bean 的外部 JAR 启动 Spring Boot 应用程序时出错
由于缺少 ServletWebServerFactory bean,Spring Boot jar 无法启动 Web 服务器