springboot项⽬突然启动缓慢的解决
⽬录
springboot项⽬突然启动缓慢
springboot启动太慢优化
1.组件⾃动扫描带来的问题(@SpringBootApplication)
2.如何避免组件⾃动扫描带来的问题(不使⽤@SpringBootApplication)
3.引发的问题——⽆法扫描组件
4.千古红楼只⼀梦,⽵篮打⽔⼀场空
5.debugdebug,bugbug更健康
6.分析Positivematches和Negativematches
7.再次优化配置信息
8.⼩结⼀下
springboot项⽬突然启动缓慢
springboot项⽬在debug模式下本来运⾏的挺快,后来某⼀天突然启动⼀半就卡在那⼀点⼀点龟速前进,还以为是我电脑问题,或者我写的代码问题,后来在⽹上搜了⼀下,结合⾃⾝项⽬情况,原来是断点问题,
有个断点⽆论如何都去不掉。可能是之前遗留的,后代码删除了,
也可能是因为这个地⽅的代码属于加载运⾏的什么节点,总之去不掉
后来根据⽹上的⽅法,在debug模式窗⼝下,选择Run菜单,点击Remove All Breakpoints的选项(好像也可以选择Skip All Breakpoing。)
然后所有断点都去掉了,重新启动,流畅
springboot启动太慢优化
接下来我们⼀起探讨下每个问题。
1.组件⾃动扫描带来的问题(@SpringBootApplication)
我们在第⼀篇博客就介绍了,我们默认情况下,我们会使⽤@SpringBootApplication注解来⾃动获取应⽤的配置信息,但这样也会带来⼀些副作⽤。使⽤这个注解后,会触发⾃动配置(auto-configuration)和组件扫描(component scanning),这跟使⽤@Configuration、@EnableAutoConfiguration和@ComponentScan三个注解的作⽤是⼀样的。这样做给开发带来⽅便的同时,会有以下的⼀些影响:
(a)会导致项⽬启动时间变长(原因:加载了我们不需要使⽤的组件,浪费了cpu资源和内存资源)。当启动⼀个⼤的应⽤程序,或将做⼤量的集成测试启动应⽤程序时,影响会特别明显。
(b)会加载⼀些不需要的多余的实例(beans)。
(c)会增加CPU消耗和内存的占⽤。
2.如何避免组件⾃动扫描带来的问题(不使⽤@ SpringBootApplication)
本着有问题就要解决的⼼态,针对以上的问题,我们要怎么解决呢?很明显,既然@SpringBootApplication加载了⼀些不必要的配置,那么我们想是否可以就加载我们⾃⼰指定的配置呢?我们的思路不使⽤@SpringBootApplication,并且不使⽤
@ComponentScan注解(此注解会⾃动扫描我们注解了@Controller,@Service的注解的类,加载到Spring IOC容器中),然后我们使⽤@Configuration和@EnableAutoConfiguration进⾏配置启动类,代码如下:
package com.kfit.spring_boot_performance;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import t.annotation.Bean;
import t.annotation.Configuration;
import com.kfit.spring_ller.HelloController;
/**
* @author Angel --守护天使
* @version v.0.1
* @date 2017年3⽉11⽇
*/
//移除 @SpringBootApplication and @ComponentScan, ⽤ @EnableAutoConfiguration 来替代
@Configuration
@EnableAutoConfiguration
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
3.引发的问题——⽆法扫描组件
我们正要为我们的代码改良庆幸的时候,我们发现问题来了。启动之后,访问我们编写的访问页⾯/index,
出现错误:There was an unexpected error (type=Not Found, status=404).
这是由于什么引起的呢?还记得我们刚刚介绍的@ComponentScan注解嘛,启⽤这个注解Spring才能够进⾏⾃动组件的扫描,否则⽆法扫描到我们编写的组件类。那么问题来了,怎么办呢?问题的解决就是:显式进⾏配置。
注⼊代码如下(假设我们写的类是HelloController,在这⾥博主直接写在App.java启动类进⾏注⼊):
@Bean
public HelloController helloController(){
return new HelloController();
}
在以上的代码中⽤ @Bean 注解明确显式配置,以便被 Spring 扫描到。
在重新启动之后,我们就可以正常访问/index页⾯了。
到这⾥肯定就会有⼈会说:那这样的话,不是会增加我们的编码量。我只能说:你既要加载快,⼜要不编码,博主实在不知道怎么办了。凡事有利有弊,⾃⼰权衡利弊。
4.千古红楼只⼀梦,⽵篮打⽔⼀场空
有⼈不相信,这个真的能启动更快吗,于是乎就编码进⾏测试。哈哈,露馅了,还是⼀样启动的跟蜗⽜⼀样慢。那为什么是这样呢?为什么我们研究了半天,最终却是:千古红楼只⼀梦,⽵篮打⽔⼀场空。
聪明的读者,会注意到我们提到:@SpringBootApplication注解的作⽤跟@EnableAutoConfiguration注解的作⽤是相当的,那就意味着它也能带来上述的问题。要避免这些问题,我们就要知道我们的组件列表是哪些?
5.debug debug,bug bug更健康
我们在上⾯说了,我们的问题就是如何知道我们的组件列表是哪些?这时候debug就隆重登场了,⿎掌欢迎debug先⽣上场。
请问debug先⽣:在此时此刻您有什么获奖感⾔?
spring boot选择题
debug先⽣:经历了慢慢⼈⽣,我终于发现我的价值了。在这⾥我要感谢CCTV、感谢MTV、感谢可⼝可乐,感谢⾮常可乐、感谢加多宝、感谢王⽼吉、感谢主办⽅SpringBoot,让我有机会在这个舞台跟⼤家见⾯。谢谢你们,我⼀定不会让⼤家失望的。
好了,废话不多说了,我们先看看如何使⽤debug呢?
第⼀种情况:使⽤spring-boot:run启动⽅式
这种情况的话,完整的运⾏代码是:
spring-boot:run -Ddebug
第⼆种情况:使⽤Run As —— Java Application启动⽅式
这种情况的话,配置VM参数即可,具体操作如下:
【右键】——【Run As】——【Run Configurations…】——【选择Arguments】——【VM arguments】中加⼊:【-Ddebug】。
这时候在启动的时候,我们就能看到控制台打印出了⼀些我们平时没看到过的⽇志信息。
=========================
AUTO-CONFIGURATION REPORT
=========================
Positive matches:
-----------------
DispatcherServletAutoConfiguration matched
- @ConditionalOnClass found required class 'org.springframework.web.servlet.DispatcherServlet'
(OnClassCondition)
- @ConditionalOnWebApplication (required) found StandardServletEnvironment (OnWebApplicationCondition)
//此处省略剩下的打印信息…
6.分析Positive matches和Negative matches
在打印信息⾥,我们有必要先了解下这⾥的⼀些知识:
(a) Positive match:累出匹配到对应类的配置项。
(b) Negative match:不包括某个配置项的原因。
现在以DataSourceAutoConfiguration举例说明:
(a)@ConditionalOnClass表⽰对应的类在classpath⽬录下存在时,才会去解析对应的配置⽂件,对于DataSourceAutoConfiguration来说就是指:只有javax.sql.DataSource和
org.springframework.bedded.EmbeddedDatabaseType类都存在时,就会配置对应的数据库资源。(b)@ConditionalOnMissingClass表⽰对应的类在classpath⽬录下不到。
(c)OnClassCondition⽤于表⽰匹配的类型(postive or negative)。
OnClassCondition是最普遍的浏览探测条件,除此之外,Spring Boot也使⽤别的探测条件,如:OnBeanCondition⽤于检测指定bean实例存在与否、OnPropertyCondition⽤于检查指定属性是否存在等等。
符合negative match代表⼀些配置类(xxxConfiguration之类的),它们虽然存在于classpath⽬录,但是修饰它们的注解中依赖的其他类不存在。
7.再次优化配置信息
根据上⾯的理论知识,我们只需要在启动的时候,显式地引⼊这些组件,拷贝Positive matches中列出的信息:DispatcherServletAutoConfiguration
EmbeddedServletContainerAutoConfiguration
ErrorMvcAutoConfiguration
HttpEncodingAutoConfiguration
HttpMessageConvertersAutoConfiguration
JacksonAutoConfiguration
JmxAutoConfiguration
MultipartAutoConfiguration
ServerPropertiesAutoConfiguration
PropertyPlaceholderAutoConfiguration
ThymeleafAutoConfiguration
WebMvcAutoConfiguration
WebSocketAutoConfiguration
然后来更新项⽬配置,显式地引⼊这些组件,引⼊之后,再运⾏⼀下应⽤确保没有错误发⽣:
@Configuration
@Import({
DispatcherServletAutoConfiguration.class,
EmbeddedServletContainerAutoConfiguration.class,
ErrorMvcAutoConfiguration.class,
HttpEncodingAutoConfiguration.class,
HttpMessageConvertersAutoConfiguration.class,
JacksonAutoConfiguration.class,
JmxAutoConfiguration.class,
MultipartAutoConfiguration.class,
ServerPropertiesAutoConfiguration.class,
PropertyPlaceholderAutoConfiguration.class,
ThymeleafAutoConfiguration.class,
WebMvcAutoConfiguration.class,
WebSocketAutoConfiguration.class,
})
public class App {
在上⾯的代码中,我们可以删掉我们不需要的组件信息,来挺⾼应⽤的性能,⽐如在项⽬中没有使⽤Jmx和WebSocket功能的话,那么我们就可以删除JmxAutoConfiguration.class和WebSocketAutoConfiguration.class。
删除掉之后,再次运⾏项⽬,确保⼀切正常。
8.⼩结⼀下
在本篇⽂章中我们介绍了如何加速spring boot快速启动,主要的思路就是废弃@SpringBootApplication显式的引⼊我们所需要的组件。
以上为个⼈经验,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论