springboot启动太慢优化
需求缘起:有⼈在【springboot】问:springboot启动慢的问题何时有个分享就好了,谢谢。粉丝的问题还是要认真的回答的。
我们先看看本节的⼤纲:
(1)组件⾃动扫描带来的问题(@SpringBootApplication);
(2)如何避免组件⾃动扫描带来的问题(不使⽤@ SpringBootApplication);
(3)引发的问题——⽆法扫描组件;
(4)千古红楼只⼀梦,⽵篮打⽔⼀场空;
(5)debug debug,bug bug更健康;
(6)分析Positive matches和Negative matches;
(7)再次优化配置信息;
(8)总结
接下来我们⼀起探讨下每个问题。
(1)组件⾃动扫描带来的问题(@SpringBootApplication);
我们在第⼀篇博客就介绍了,我们默认情况下,我们会使⽤@SpringBootApplication注解来⾃动获取应⽤的配置信息,但这样也会带来⼀些副作⽤。使⽤这个注解后,会触发⾃动配置(auto-configuration)和组件扫描(component scanning),这跟使⽤@Configuration、@EnableAutoConfiguration和@ComponentScan三个注解的作⽤是⼀样的。这样做给开发带来⽅便的同时,会有以下的⼀些影响:
(a)会导致项⽬启动时间变长(原因:加载了我们不需要使⽤的组件,浪费了cpu资源和内存资源)。当启动⼀个⼤的应⽤程序,或将做⼤量的集成测试启动应⽤程序时,影响会特别明显。
(b)会加载⼀些不需要的多余的实例(beans)。
(c)会增加CPU消耗和内存的占⽤。
(2)如何避免组件⾃动扫描带来的问题(不使⽤@ SpringBootApplication);
本着有问题就要解决的⼼态,针对以上的问题,我们要怎么解决呢?很明显,既然@SpringBootAppli
cation加载了⼀些不必要的配置,那么我们想是否可以就加载我们⾃⼰指定的配置呢?我们的思路不使⽤@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先⽣:在此时此刻您有什么获奖感⾔?
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,spring ioc注解
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显式的引⼊我们所需要的组件。
下节预告:介绍⾼性能Web服务器Undertow,在下⼀篇介绍如何替换Tomcat使⽤Undertow进⾏内存优化。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论