SpringBoot 是依赖于 Spring 的,比起 Spring,除了拥有 Spring 的全部功能以外,SpringBoot 无需繁琐的 Xml 配置,这取决于它自身强大的自动装配功能;并且自身已嵌入Tomcat、Jetty 等 web 容器,集成了 SpringMvc,使得 SpringBoot 可以直接运行,不需要额外的容器,提供了一些大型项目中常见的非功能性特性,如嵌入式服务器、安全、指标,健康检测、外部配置等,
其实 Spring 大家都知道,Boot 是启动的意思。所以,Spring Boot 其实就是一个启动 Spring 项目的一个工具而已,总而言之,SpringBoot 是一个服务于框架的框架;也可以说 SpringBoot 是一个工具,这个工具简化了 Spring 的配置;
SpringBoot的启动经过了一些一系列的处理,我们先看看整体过程的流程图

可以肯定的是,所有的标准的 SpringBoot 的应用程序都是从 run 方法开始的
package com.spring;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
@SpringBootApplication
public class App {
public static void main(String[] args) {
// 启动springboot
ConfigurableApplicationContext run = SpringApplication.run(App.class, args);
}
}
进入 run 方法后,会 new 一个 SpringApplication 对象,创建这个对象的构造函数做了一些准备工作,编号第 2~5 步就是构造函数里面所做的事情
/**
* Static helper that can be used to run a {@link SpringApplication} from the
* specified sources using default settings and user supplied arguments.
* @param primarySources the primary sources to load
* @param args the application arguments (usually passed from a Java main method)
* @return the running {@link ApplicationContext}
*/
public static ConfigurableApplicationContext run(Class<?>[] primarySources,
String[] args) {
return new SpringApplication(primarySources).run(args);
}
在 SpringApplication 的构造方法内,首先会通过 WebApplicationType.deduceFromClasspath(); 方法判断当前应用程序的容器,默认使用的是Servlet 容器,除了 Servlet 之外,还有 NONE 和 REACTIVE (响应式编程);

具体代码
public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
// resourceLoader 赋值为 Null
this.resourceLoader = resourceLoader;
// primarySources不为空,继续向下执行。为空抛异常
Assert.notNull(primarySources, "PrimarySources must not be null");
// 将 SpringbootdemoApplication(启动类)赋值给 primarySources
this.primarySources = new LinkedHashSet<>(Arrays.asList(primarySources));
// 从 classpath 类路径推断 Web 应用类型,有三种 Web 应用类型,分别是
// NONE: 该应用程序不应作为 Web 应用程序运行,也不应启动嵌入式 Web 服务器
// SERVLET: 该应用程序应作为基于 servlet 的 Web 应用程序运行,并应启动嵌入式 servlet Web 服务器。
// REACTIVE: 该应用程序应作为响应式 Web 应用程序运行,并应启动嵌入式响应式 Web 服务器
this.webApplicationType = WebApplicationType.deduceFromClasspath();
// 初始化 bootstrapRegistryInitializers,通过 getSpringFactoriesInstances()获取工厂实例,
// 底层使用的是反射Class<?> instanceClass = ClassUtils.forName(name, classLoader)动态加载实例对象。
this.bootstrapRegistryInitializers = new ArrayList<>(
getSpringFactoriesInstances(BootstrapRegistryInitializer.class));
// 初始化 ApplicationContextInitializer 集合
setInitializers((Collection) getSpringFactoriesInstances(ApplicationContextInitializer.class));
// 初始化 ApplicationListener
setListeners((Collection) getSpringFactoriesInstances(ApplicationListener.class));
// 获取 StackTraceElement 数组遍历,通过反射获取堆栈中有 main 方法的类
this.mainApplicationClass = deduceMainApplicationClass();
}
这里加载的初始化器是 SpringBoot 自带初始化器,从从 META-INF/spring.factories 配置文件中加载的,那么这个文件在哪呢?自带有 2 个,分别在源码的 Jar 包的 spring-boot-autoconfigure 项目 和 spring-boot 项目里面各有一个

spring.factories文件里面,看到开头是 org.springframework.context.ApplicationContextInitializer 接口就是初始化器了

我们也可以自己实现一个自定义的初始化器:实现 ApplicationContextInitializer 接口既可
MyApplicationContextInitializer.java
package com.spring.application;
import org.springframework.context.ApplicationContextInitializer;
import org.springframework.context.ConfigurableApplicationContext;
/**
* 自定义的初始化器
*/
public class MyApplicationContextInitializer implements ApplicationContextInitializer<ConfigurableApplicationContext> {
@Override
public void initialize(ConfigurableApplicationContext configurableApplicationContext) {
System.out.println("我是初始化的 MyApplicationContextInitializer...");
}
}
在 resources 目录下添加 META-INF/spring.factories 配置文件,内容如下,将自定义的初始化器注册进去;
org.springframework.context.ApplicationContextInitializer=\
com.spring.application.MyApplicationContextInitializer

启动 SpringBoot 后,就可以看到控制台打印的内容了,在这里我们可以很直观的看到它的执行顺序,是在打印 banner的后面执行的;

加载监听器也是从 META-INF/spring.factories 配置文件中加载的,与初始化不同的是,监听器加载的是实现了 ApplicationListener 接口的类

自定义监听器也跟初始化器一样,依葫芦画瓢就可以了,这里不在举例;
deduceMainApplicationClass(); 这个方法仅仅是找到 main 方法所在的类,为后面的扫包作准备,deduce 是推断的意思,所以准确地说,这个方法作用是推断出主方法所在的类;

程序运行到这里,就已经进入了 run 方法的主体了,第一步调用的 run 方法是静态方法,那个时候还没实例化 SpringApplication 对象,现在调用的 run 方法是非静态的,是需要实例化后才可以调用的,进来后首先会开启计时器,这个计时器有什么作用呢?顾名思义就使用来计时的嘛,计算 SpringBoot 启动花了多长时间;关键代码如下:
// 实例化计时器
StopWatch stopWatch = new StopWatch();
// 开始计时
stopWatch.start();
run 方法代码段截图

这里将 java.awt.headless 设置为 true,表示运行在服务器端,在没有显示器器和鼠标键盘的模式下照样可以工作,模拟输入输出设备功能。
做了这样的操作后,SpringBoot 想干什么呢?其实是想设置该应用程序,即使没有检测到显示器,也允许其启动。对于服务器来说,是不需要显示器的,所以要这样设置
方法主体如下:
private void configureHeadlessProperty() {
System.setProperty(SYSTEM_PROPERTY_JAVA_AWT_HEADLESS, System.getProperty(
SYSTEM_PROPERTY_JAVA_AWT_HEADLESS, Boolean.toString(this.headless)));
}
通过方法可以看到,setProperty() 方法里面又有个 getProperty();这不多此一举吗?其实 getProperty() 方法里面有2个参数, 第一个 key值,第二个是默认值,意思是通过 key 值查找属性值,如果属性值为空,则返回默认值 true;保证了一定有值的情况;
这一步 通过监听器来实现初始化的的基本操作,这一步做了 2 件事情

将执行 run 方法时传入的参数封装成一个对象

仅仅是将参数封装成对象,没啥好说的,对象的构造函数如下
public DefaultApplicationArguments(String[] args) {
Assert.notNull(args, "Args must not be null");
this.source = new Source(args);
this.args = args;
}
那么问题来了,这个参数是从哪来的呢?其实就是 main 方法里面执行静态 run 方法传入的参数

准备环境变量,包含系统属性和用户配置的属性,执行的代码块在 prepareEnvironment 方法内

打了断点之后可以看到,它将 maven 和系统的环境变量都加载进来了

这个方法 configureIgnoreBeanInfo() 这个方法是将 spring.beaninfo.ignore 的默认值值设为 true,意思是跳过 beanInfo 的搜索,其设置默认值的原理和第 7 步一样;
private void configureIgnoreBeanInfo(ConfigurableEnvironment environment) {
if (System.getProperty(
CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME) == null) {
Boolean ignore = environment.getProperty("spring.beaninfo.ignore",
Boolean.class, Boolean.TRUE);
System.setProperty(CachedIntrospectionResults.IGNORE_BEANINFO_PROPERTY_NAME,
ignore.toString());
}
}
当然也可以在配置文件中添加以下配置来设为 false
spring.beaninfo.ignore=false
显而易见,这个流程就是用来打印控制台那个很大的 Spring 的 Banner 的,就是下面这个东东

那他在哪里打印的呢?他在 SpringBootBanner.java 里面打印的,这个类实现了 Banner 接口,
而且 Banner 信息是直接在代码里面写死的;

有些公司喜欢自定义 Banner 信息,如果想要改成自己喜欢的图标该怎么办呢,其实很简单,只需要在 resources 目录下添加一个 banner.txt 的文件即可,txt 文件内容如下
,-----.
| |) /_ ,---. ,--,--.,--.--.
| .-. \| .-. :' ,-. || .--'
| '--' /\ --.\ '-' || |
`------' `----' `--`--'`--'
一定要添加到 resources 目录下,别加错了

只需要加一个文件即可,其他什么都不用做,然后直接启动 SpringBoot,就可以看到效果了
实例化应用程序的上下文, 调用 createApplicationContext() 方法,这里就是用反射创建对象,没什么好说的;

异常报告器是用来捕捉全局异常使用的,当 SpringBoot 应用程序在发生异常时,异常报告器会将其捕捉并做相应处理,在 spring.factories 文件里配置了默认的异常报告器

需要注意的是,这个异常报告器只会捕获启动过程抛出的异常,如果是在启动完成后,在用户请求时报错,异常报告器不会捕获请求中出现的异常

了解原理了,接下来我们自己配置一个异常报告器来玩玩;
MyExceptionReporter.java 继承 SpringBootExceptionReporter 接口
package com.spring.application;
import org.springframework.boot.SpringBootExceptionReporter;
import org.springframework.context.ConfigurableApplicationContext;
public class MyExceptionReporter implements SpringBootExceptionReporter {
private ConfigurableApplicationContext context;
// 必须要有一个有参的构造函数,否则启动会报错
MyExceptionReporter(ConfigurableApplicationContext context) {
this.context = context;
}
@Override
public boolean reportException(Throwable failure) {
System.out.println("进入异常报告器");
failure.printStackTrace();
// 返回false会打印详细springboot错误信息,返回true则只打印异常信息
return false;
}
}
在 spring.factories 文件中注册异常报告器
# Error Reporters 异常报告器
org.springframework.boot.SpringBootExceptionReporter=\
com.spring.application.MyExceptionReporter

接着我们在 application.yml 中 把端口号设置为一个很大的值,这样肯定会报错
server:
port: 80828888
启动后,控制台打印如下图

这里准备的上下文环境是为了下一步刷新做准备的,里面还做了一些额外的事情;

在 postProcessApplicationContext(context); 方法里面。使用单例模式创建 了BeanNameGenerator 对象,其实就是 beanName 生成器,用来生成 bean 对象的名称
初始化方法有哪些呢?还记得第 3 步里面加载的初始化器嘛?其实是执行第3步加载出来的所有初始化器,实现 ApplicationContextInitializer 接口的类
这里将启动参数以单例的模式注册到容器中,是为了以后方便拿来使用,参数的beanName 为 :springApplicationArguments
刷新上下文已经是 Spring 的范畴了,自动装配和启动 tomcat 就是在这个方法里面完成的,还有其他的 Spring 自带的机制在这里就不一一细说了

afterRefresh 方法是启动后的一些处理,留给用户扩展使用,目前这个方法里面是空的
/**
* Called after the context has been refreshed.
* @param context the application context
* @param args the application arguments
*/
protected void afterRefresh(ConfigurableApplicationContext context,
ApplicationArguments args) {
}
到这一步,Springboot 其实就已经完成了,计时器会打印启动 SpringBoot 的时长

在控制台看到启动还是挺快的,不到 2 秒就启动完成了;

告诉应用程序,我已经准备好了,可以开始工作了

执行自定义的 run 方法
这是一个扩展功能,callRunners(context, applicationArguments) 可以在启动完成后执行自定义的 run 方法;有 2 种方式可以实现:
接下来我们验证一把,为了方便代码可读性,我把这 2 种方式都放在同一个类里面
package com.spring.init;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;
/**
* 自定义run方法的2种方式
*/
@Component
public class MyRunner implements ApplicationRunner, CommandLineRunner {
@Override
public void run(ApplicationArguments args) throws Exception {
System.out.println(" 我是自定义的run方法1,实现 ApplicationRunner 接口既可运行" );
}
@Override
public void run(String... args) throws Exception {
System.out.println(" 我是自定义的run方法2,实现 CommandLineRunner 接口既可运行" );
}
}
启动 SpringBoot 后就可以看到控制台打印的信息了

具体代码
public ConfigurableApplicationContext run(String... args) {
long startTime = System.nanoTime();
// 通过 BootstrapRegistryInitializer来initialize 默认的 DefaultBootstrapContext
DefaultBootstrapContext bootstrapContext = createBootstrapContext();
ConfigurableApplicationContext context = null;
// 配置java.awt.headless属性
configureHeadlessProperty();
// 获取 SpringApplicationRunListeners 监听器
SpringApplicationRunListeners listeners = getRunListeners(args);
// 启动 SpringApplicationRunListeners 监听,表示 SpringApplication 启动(触发ApplicationStartingEvent事件)
listeners.starting(bootstrapContext, this.mainApplicationClass);
try {
// 创建 ApplicationArguments 对象,封装了 args 参数
ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);
// 做相关环境准备,绑定到 SpringApplication,返回可配置环境对象ConfigurableEnvironment
ConfigurableEnvironment environment = prepareEnvironment(listeners, bootstrapContext, applicationArguments);
// 配置 spring.beaninfo.ignore,设置为 true.即跳过搜索Bean信息
configureIgnoreBeanInfo(environment);
// 控制台打印 SpringBoot 的 Banner(横幅)标志
Banner printedBanner = printBanner(environment);
// 根据WebApplicationType从ApplicationContextFactory工厂创建ConfigurableApplicationContext
context = createApplicationContext();
// 设置ConfigurableApplicationContext中的ApplicationStartup为DefaultApplicationStartup
context.setApplicationStartup(this.applicationStartup);
// 应用所有的ApplicationContextInitializer容器初始化器初始化context,触发ApplicationContextInitializedEvent事件监听,打印启动日志信息,启动Profile日志信息
// ConfigurableListableBeanFactory中注册单例Bean(springApplicationArguments),并为该BeanFactory中的部分属性赋值。
// 加载所有的source.并将Bean加载到ConfigurableApplicationContext,触发ApplicationPreparedEvent事件监听
prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
// 刷新容器(在方法中集成了Web容器具体请看 https://editor.csdn.net/md/?articleId=123136262)
refreshContext(context);
// 刷新容器的后置处理(空方法)
afterRefresh(context, applicationArguments);
// 启动花费的时间
Duration timeTakenToStartup = Duration.ofNanos(System.nanoTime() - startTime);
if (this.logStartupInfo) {
// 打印日志Started xxx in xxx seconds (JVM running for xxxx)
new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), timeTakenToStartup);
}
// 触发 ApplicationStartedEvent 事件监听。上下文已刷新,应用程序已启动。
listeners.started(context, timeTakenToStartup);
// 调用 ApplicationRunner和CommandLineRunner
callRunners(context, applicationArguments);
}
// 处理运行时发生的异常,触发 ApplicationFailedEvent 事件监听
catch (Throwable ex) {
handleRunFailure(context, ex, listeners);
throw new IllegalStateException(ex);
}
try {
// 启动准备消耗的时间
Duration timeTakenToReady = Duration.ofNanos(System.nanoTime() - startTime);
// 在 run 方法完成前立即触发 ApplicationReadyEvent事件监听,表示应用上下文已刷新,并且CommandLineRunners和ApplicationRunners已被调用。
listeners.ready(context, timeTakenToReady);
}
catch (Throwable ex) {
handleRunFailure(context, ex, null);
throw new IllegalStateException(ex);
}
return context;
}
转载请标明出处,原文地址:https://blog.csdn.net/weixin_41835916 如果觉得本文对您有帮助,请点击赞支持一下,您的支持是我写作最大的动力,谢谢。
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www
我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我
什么是ruby的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht
一、引擎主循环UE版本:4.27一、引擎主循环的位置:Launch.cpp:GuardedMain函数二、、GuardedMain函数执行逻辑:1、EnginePreInit:加载大多数模块int32ErrorLevel=EnginePreInit(CmdLine);PreInit模块加载顺序:模块加载过程:(1)注册模块中定义的UObject,同时为每个类构造一个类默认对象(CDO,记录类的默认状态,作为模板用于子类实例创建)(2)调用模块的StartUpModule方法2、FEngineLoop::Init()1、检查Engine的配置文件找出使用了哪一个GameEngine类(UGame
这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/
HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候
遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg
我基本上来自Java背景并且努力理解Ruby中的模运算。(5%3)(-5%3)(5%-3)(-5%-3)Java中的上述操作产生,2个-22个-2但在Ruby中,相同的表达式会产生21个-1-2.Ruby在逻辑上有多擅长这个?模块操作在Ruby中是如何实现的?如果将同一个操作定义为一个web服务,两个服务如何匹配逻辑。 最佳答案 在Java中,模运算的结果与被除数的符号相同。在Ruby中,它与除数的符号相同。remainder()在Ruby中与被除数的符号相同。您可能还想引用modulooperation.