springboot项目突然启动缓慢的解决

目录
  • springboot项目突然启动缓慢
  • springboot启动太慢优化
    • 1.组件自动扫描带来的问题(@SpringBootApplication)
    • 2.如何避免组件自动扫描带来的问题(不使用@ SpringBootApplication)
    • 3.引发的问题——无法扫描组件
    • 4.千古红楼只一梦,竹篮打水一场空
    • 5.debug debug,bug bug更健康
    • 6.分析Positive matches和Negative matches
    • 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 org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import com.kfit.spring_boot_performance.controller.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.jdbc.datasource.embedded.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显式的引入我们所需要的组件。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • springboot项目启动慢的问题排查方式

    目录 springboot项目启动慢的问题排查 1.最开始查看的启动日志,是在输出: 2. 启动项目,打印日志级别改为debug,查看更详细信息 如何优化SpringBoot的项目的启动速度 实际上它是下面三个注解的组合 可以用@SpringBootApplication注解下面的属性 springboot项目启动慢的问题排查 springboot项目,随着时间的推移,启动耗时逐步增加,从几分钟慢慢的达到30多分钟,有点恐怖! 项目中用到技术:hibernate.redis.kafka.线程池等

  • springboot程序启动慢-未配置hostname的解决

    目录 springboot程序启动慢-未配置hostname 具体日志如下 通过jstack查看线程发现如下信息 查看源码如下 Spring Boot启动速度慢的原因总结 前提条件 原因一 原因二 原因三 解决方法 springboot程序启动慢-未配置hostname 今天有同时反馈自己的程序在本地mac环境还正常,在生产环境上centos7启动特别慢 具体日志如下 可以看到程序在这里卡了将近2分钟时间,期间无任何日志输出. 通过jstack查看线程发现如下信息 查看源码如下 public n

  • 解决Idea启动Spring Boot很慢的问题

    同事win10启动Spring Boot工程只要3秒左右,我的启动要30秒.开始以为是CPU差距太大,后来才觉得不是这样! 解决方案 1. hostname命令查看自己的 hostname $ hostname 返回类似如下信息 aoedeMacBook-Pro.local 2. 修改host文件 sudo vim /etc/hosts 修改条目如下,注意分隔符是TAB!不是空格 127.0.0.1 localhost aoedeMacBook-Pro.local ::1 localhost a

  • springboot项目突然启动缓慢的解决

    目录 springboot项目突然启动缓慢 springboot启动太慢优化 1.组件自动扫描带来的问题(@SpringBootApplication) 2.如何避免组件自动扫描带来的问题(不使用@ SpringBootApplication) 3.引发的问题--无法扫描组件 4.千古红楼只一梦,竹篮打水一场空 5.debug debug,bug bug更健康 6.分析Positive matches和Negative matches 7.再次优化配置信息 8.小结一下 springboot项目

  • springboot 项目容器启动后如何自动执行指定方法

    目录 springboot 项目容器启动后自动执行指定 springboot 容器及启动过程 问题1:为什么要启动父子两个容器? 问题2:在什么时候启动父容器? 问题3:父容器和子容器的区别? 问题4:怎么保证父容器启动过程中 问题5:容器实际通过什么来管理bean springboot 项目容器启动后自动执行指定 我们需要写一个class去让它实现ApplicationRunner,然后重写它的run方法就行了,注意类加上@Component注解 注意:这个class需要写在和Applicat

  • idea新建maven项目时速度缓慢的解决方法

    原因 IDEA根据maven archetype的本质,其实是执行mvn archetype:generate命令,该命令执行时,需要指定一个archetype-catalog.xml文件. 该命令的参数-DarchetypeCatalog,可选值为:remote,internal  ,local等,用来指定archetype-catalog.xml文件从哪里获取. 默认为remote,即从 http://repo1.maven.org/maven2/archetype-catalog.xml路

  • 解决复制springboot项目后,启动日志无颜色的问题

    复制springboot项目后,启动日志无颜色 把之前的springboot项目复制到idea后,启动日志无颜色,这是因为复制过来的项目并没有选择springboot模板,需要做下简单的修改. 1,问题图片如下,虽然不影响开发,但是看着就是不爽,改他. 2,点击工具栏的启动设置,如下图: 3,点击左上角"+",然后选择下面的springboot模板 4,选择启动类,然后apply即可.上面的Name随意定义,可以用项目名字 5,重新启动 以上为个人经验,希望能给大家一个参考,也希望大家

  • 测试springboot项目出现Test Ignored的解决

    目录 测试springboot项目出现Test Ignored 测试类不报错只提示Test ignored的解决 测试springboot项目出现Test Ignored 今天在写springBoot项目运行测试类时出现了以下问题: Test ignored. java.lang.IllegalStateException: Unable to find a @SpringBootConfiguration, you need to use @ContextConfiguration or @S

  • 超个性修改SpringBoot项目的启动banner的方法

    如果我们使用过SpringBoot,那么就会对下面的图案不陌生.Springboot 启动的同时会打印下面的图案,并带有版本号. 查看SpringBoot官方文档可以找到关于 banner 的描述 The banner that is printed on start up can be changed by adding a banner.txt file to your classpath or by setting the spring.banner.location property t

  • springboot项目启动的时候参数无效的解决

    目录 springboot项目启动的时候参数无效 改动run方法的参数,添加args参数如下 springboot项目启动参数设置问题 Spring boot项目常用的几种启动方式 war包部署方式 springboot项目启动的时候参数无效 今天启动一个springboot项目发现启动的时候输入的参数都是不能生效,但是yaml文件的配置却生效了,排查了半天,最后发现原来启动类里面有问题,原代码如下 public class Application { public static void ma

  • springboot项目拦截器重定向循环问题的解决

    目录 springboot项目拦截器重定向循环 解决办法 springboot拦截器无限循环报错 springboot项目拦截器重定向循环 本菜鸟很久没写东西了,这回是解决了一个小问题,希望能帮助到你. 最近写了一个项目,项目中写了登录拦截器,session过期的用户请求会重定向到登录页面. 写完测试发现session过期后浏览器一直提示重定向次数过多. 打开浏览器F12看到一直都在访问/login/login/login之类的无限循环路径. 后来经过网上查询发现我之前设置重定向地址时直接设置的

  • SpringBoot项目修改访问端口和访问路径的方法

    创建SpringBoot项目,启动后,默认的访问路径即主机IP+默认端口号8080:http://localhost:8080/ 此时,我们就可以访问Controller层的接口了,如:http://localhost:8080/hello package com.springboot.test; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.a

  • springBoot项目启动类启动无法访问的解决方法

    网上也查了一些资料,我这里总结.下不来虚的,也不废话. 解决办法: 1.若是maven项目,则找到右边Maven Projects --->Plugins--->run(利用maven启动)则可以加载到webapp资源 2.上面方法治标不治本.在项目的pom文件中添加<bulid>标签标注路径即可,pom.xml后部分代码如下: 刷新maven加载,重启项目.若还是无法访问,重新导入项目 <dependencies> xxxxxxxxxxxx </dependen

随机推荐