Please set spring.main.web-application-type=reactive or remove spring-boot-starter-web dependency.
时间:2024-04-06 16:15:26 来源:网络cs 作者:璐璐 栏目:卖家故事 阅读:
一、问题
在启动springcloud的gateway模块的时候报错Please set spring.main.web-application-type=reactive or remove spring-boot-starter-web dependency.
二、问题产生的原因
gateway组件中的 spring-boot-starter-webflux 和 springboot作为web项目启动必不可少的 spring-boot-starter-web 出现冲突。
三、解决方案(任选一种就可以)
3.1 注释pom.xml内容
在gateway的pom文件上注释掉spring-boot-starter-web代码
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency>
3.2修改配置文件
在配置文件上加上
spring: main: web-application-type: reactive
解释:这里面涉及到springboot的启动流程因为spring-boot-starter-webflux包,在springboot启动类型里面表示的就是reactive。我们可以看源码,了解一下springboot启动流程
一步一步点进去
这是springApplication构造方法,我们先看构造方法
点进去
3.2.1通过springboot源码解释
springApplication构造源码(不同版本会有差异,当前版本是2.6.1)
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) { this.sources = new LinkedHashSet(); // 在控制台打印banner.txt文本内容 this.bannerMode = Mode.CONSOLE; this.logStartupInfo = true; this.addCommandLineProperties = true; this.addConversionService = true; this.headless = true; this.registerShutdownHook = true; this.additionalProfiles = Collections.emptySet(); this.isCustomEnvironment = false; this.lazyInitialization = false; this.applicationContextFactory = ApplicationContextFactory.DEFAULT; this.applicationStartup = ApplicationStartup.DEFAULT; // 开始输resourceLoader注入了属性null this.resourceLoader = resourceLoader; Assert.notNull(primarySources, "PrimarySources must not be null"); //将启动类从数组重新封装Set,注入到primarySources当中 this.primarySources = new LinkedHashSet(Arrays.asList(primarySources)); /** *webApplicationType有三种类型,REACTIVE,SERVLET,NONE * 引入spring-boot-starter-web就是SERVLET * 引入spring-boot-starter-webflux就是REACTIVE * 没有就是NONE */ this.webApplicationType = WebApplicationType.deduceFromClasspath(); /** *从spring-cloud-context的jar包的META-INF/spring.factories文件中得到key为org.springfra *mework.boot.BootstrapRegistryInitializer的全类名集合,进行实例化,然后注入 bootstrapRegi *stryInitializers 属性,其中核心方法getSpringFactoriesInstances *下面有getSpringFactoriesInstances方法源码详解 *这里简单的解释一下getSpringFactoriesInstances方法就是从META-INF/spring.factories读取配 *置文件 */ this.bootstrapRegistryInitializers = new ArrayList(this.getSpringFactoriesInstances(BootstrapRegistryInitializer.class)); /** *依然调用getSpringFactoriesInstances方法 *从spring-boot的jar包的META-INF/spring.factories文件中得到key为org.springframework. *context.ApplicationContextInitializer的全类名集合,然后进行实例化,然后注入initializers *(初始化容器集合)属性 */ this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class)); //跟上面一样,这里是得到监听器的集合,并注入 this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class)); //获取当前的main方法运行的类,也就是我们的主类 this.mainApplicationClass = this.deduceMainApplicationClass(); }
上面说的META-INF/spring.factories都是在springboot.jar包中,不过BootstrapRegistryInitializer是在spring-cloud-context中。
3.2.2了解getSpringFactoriesInstances方法
随便选一个点进去
private <T> Collection<T> getSpringFactoriesInstances(Class<T> type, Class<?>[] parameterTypes, Object... args) { ClassLoader classLoader = this.getClassLoader(); /** SpringFactoriesLoader.loadFactoryNames(type, classLoader)这里才是核心, *这个方法会扫描所有jar包类路径下 META-INF/spring.factories */ Set<String> names = new LinkedHashSet(SpringFactoriesLoader.loadFactoryNames(type, classLoader)); List<T> instances = this.createSpringFactoriesInstances(type, parameterTypes, classLoader, args, names); AnnotationAwareOrderComparator.sort(instances); return instances; }
阅读本书更多章节>>>>
本文链接:https://www.kjpai.cn/gushi/2024-04-06/154876.html,文章来源:网络cs,作者:璐璐,版权归作者所有,如需转载请注明来源和作者,否则将追究法律责任!
上一篇:论文ai率怎么降,其实不难
下一篇:返回列表