四川建设网站信息查询中心,企业门户网站开发要多少钱,推广链接怎么自己搞定,网站备案手续通过深入分析SpringBoot中WebServer的启动流程#xff0c;插入自定义的Loading页面展示逻辑#xff0c;优化软件使用时的用户体验。 背景
Java本身的特点#xff0c;再加上开发人员能力差#xff0c;软件开发工程化程度低等等问题#xff0c;经过一段时间的迭代之后… 通过深入分析SpringBoot中WebServer的启动流程插入自定义的Loading页面展示逻辑优化软件使用时的用户体验。 背景
Java本身的特点再加上开发人员能力差软件开发工程化程度低等等问题经过一段时间的迭代之后经常会出现的一个问题就是应用启动越来越慢。
过往我也是收集了不少相关资料例如 云音乐服务端应用启动时间下降40%实践分享7min到40s SpringBoot 启动优化实践等等。
本文跳出与上述文章类似的技术角度借鉴Jenkins的启动流程 —— Jenkins也是Java开发的而且其启动速度虽然经过不少优化但依然有着非常明显的延迟但Jenkins通过加入Loading页面的方式极大提升了用户在等待系统就绪期间的用户体验 —— 启动之后马上呈现给用户一个Loading页面待后台服务能够正常提供服务之后冲向到常规Index页面。
实现
Jenkins默认适用的是winstone 容器所以我在一开始所设想的拿来主义被掐死在了摇篮里最后经过一番摸索基于SpringBoot Undertow实现了类似的效果。
思路
SpringBoot项目中单单一个WebServer的启动是很快的只是SpringBoot的启动流程里其会在WebServer启动之前初始化所有的Bean而这是整个启动流程里最耗时的。所以我们将在SpringBoot容器中的Bean实例化之前启动我们自定义的minimal Undertow Server来向用户提供对于 loading页面的响应。待Spring容器准备进行WebServer的启动时停止我们的minimal Undertow Server以避免端口占用。前端loading页面将定期轮询后端服务启动情况待其正常响应时跳转到相应的主体服务页面。
直接上代码
// 1. 应用启动入口类// 应用启动入口public static void main(String[] args) {final SpringApplication springApplication new SpringApplication(SpringBootTestApplication.class);// 读取用户配置, 决定启动方式final Boolean humanableStart Convert.toBool(CommonUtil.getProperty(START_HUMANABLE, false));if (humanableStart) {springApplication.setApplicationContextClass(AnnotationConfigServletWebServerApplicationContextEx.class); // 自定义}// 启动Spring容器springApplication.run(args);}// 2. SpringBoot不同版本实现方式不同
// 这里以 SpringBoot 2.3.3为例, 更高版本需要实现 ApplicationContextFactory 接口
class AnnotationConfigServletWebServerApplicationContextWithHumanableStart extends AnnotationConfigServletWebServerApplicationContext {Undertow minimalUndertowserver;Overrideprotected void prepareRefresh() {// 尽量提前minimal Undertow Server的创建, 优化用户体验.String property this.getEnvironment().getProperty(server.port);minimalUndertowserver minimalUndertowserver(Convert.toInt(property));super.prepareRefresh();}Overrideprotected void onRefresh() {// ServletWebServerApplicationContext正是通过覆写本方法来实现 WebServer 创建的// 同时会向容器中注入webServerStartStop Bean借助Spring的生命周期回调接口SmartLifecycle来负责将webServer的开启和关闭;super.onRefresh();}Overrideprotected void finishRefresh() {// 关键流程: AbstractApplicationContext.refresh()// super.finishRefresh()中将触发 WebServerStartStopLifecycle.start() 以启动webserver, 所以我们得在它之前将我们的轻量级webserver关闭掉.minimalUndertowserver.stop();super.finishRefresh();}static Undertow minimalUndertowserver(int port) {// 这个loading.html是从jenkins里扒过来了, 也算是实现了部分拿来主义final String loadingHtml ResourceUtil.readStr(static/loading.html, CharsetUtil.CHARSET_UTF_8);// Start the minimal Undertow serverUndertow undertow Undertow.builder().addHttpListener(port, 0.0.0.0).setHandler(new HttpHandler() {Overridepublic void handleRequest(HttpServerExchange exchange) throws Exception {exchange.getResponseHeaders().put(Headers.CONTENT_TYPE, text/html);exchange.getResponseSender().send(loadingHtml);}}).build();undertow.start();return undertow;}
}原理分析
Spring容器之所以启动慢主要原因肯定就是各类Bean的实例化耗时叠加
// AbstractApplicationContext.java
// Spring核心启动流程
Override
public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// Prepare this context for refreshing.prepareRefresh(); // 我们在这里启动 minimal Undertow Server// Tell the subclass to refresh the internal bean factory.ConfigurableListableBeanFactory beanFactory obtainFreshBeanFactory();// Prepare the bean factory for use in this context.prepareBeanFactory(beanFactory);try {// Allows post-processing of the bean factory in context subclasses.postProcessBeanFactory(beanFactory);// Invoke factory processors registered as beans in the context.invokeBeanFactoryPostProcessors(beanFactory);// Register bean processors that intercept bean creation.registerBeanPostProcessors(beanFactory);// Initialize message source for this context.initMessageSource();// Initialize event multicaster for this context.initApplicationEventMulticaster();// Initialize other special beans in specific context subclasses.onRefresh();// Check for listener beans and register them.registerListeners();// Instantiate all remaining (non-lazy-init) singletons.finishBeanFactoryInitialization(beanFactory);// Last step: publish corresponding event.finishRefresh(); // 在这里关闭 minimal Undertow Server.} catch (BeansException ex) {if (logger.isWarnEnabled()) {logger.warn(Exception encountered during context initialization - cancelling refresh attempt: ex);}// Destroy already created singletons to avoid dangling resources.destroyBeans();// Reset active flag.cancelRefresh(ex);// Propagate exception to caller.throw ex;}finally {// Reset common introspection caches in Springs core, since we// might not ever need metadata for singleton beans anymore...resetCommonCaches();}}
}相关
Jenkins源码 - loading页面Jenkins源码 - loading后端代码GitHub - jenkinsci-winstone