答案:
SpringBoot的用来快速开发Spring应用的一个脚手架、其设计目的是用来简化新spring应用的初始搭建以及开发过程。
答案:
SpringBoot 它不是一个框架、它是一个可以快速构建基于Spring的脚手架(里面包含了Spring和各种框架),为开发Spring生态其他框架铺平道路二者不是一个层面的东西,没有可比性。
答案:
@SpringBootApplication:常用于启动类,标记这个应用是一个SpringBoot应用

@SpringBootConfiguration:这个注解其实就是@Configuration,表示启动类是一个配置类
@EnableAutoConfiguration:向Spring容器中导入一个Selector,用来加载类路径下的SpringFactories中定义的自动配置类,并将这些自动加载为配置Bean

此外,SpringBoot还有一些注解,用于进行定制开发(当满足什么条件时才启用):
@ConditionalOnBean
@ConditionalOnMissingBean
@ConditionalOnClass
@ConditionalOnExpression
@ConditionalOnMissingBean
…

答案:
@EnableAutoConfiguration (负责启动自动配置功能)@lmportDeferredlmportSelector,它会使SpringBoot的自动配置类的顺序在最后,这样方便我们扩展和覆盖/META-INF/spring.factories文件普通的jar包直接运行,报错:jar包中没有主清单属性

答案:
SpringBoot提供了一个插件spring-boot-maven-plugin,用于把程序打包成一个可执行的jar包
SpringBoot应用打包后,生成一个Fat jar(jar包中包含jar),包含了应用依赖的jar包和SpringBoot loader相关的类

java -jar命令执行时,会去jar中找manifest文件,在那里面找到真正的启动类


Fat jar的启动Main方法是JarLauncher,它负责创建一个LaunchedURLClassLoader来加载boot-lib下面所依赖的那些jar,并以一个新线程启动应用的Main方法(找到manifest文件中的Start-Class)





此问题即run方法背后在做些什么?
答案:
运行main方法:初始化SpringApplication,从spring.factories中读取listener、ApplicationContextInitializer

运行run方法
读取环境变量、配置信息…
创建SpringApplication上下文:ServletWebServerApplicationContext
预初始化上下文 :读取启动类,将启动类做为配置类读取,注册为BeanDefinition
调用refresh方法加载IoC容器
在这个过程中springboot会调用很多监听器对外进行扩展
答案:
ServletWebServerFactoryAutoConfiguration servlet容器自动配置类先说下实现:
<packaging>war</packaging>

<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-tomcatartifactId>
<scope>providedscope>
dependency>
public class ServletInitializer extends SpringBootServletInitializer {
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
// 传入SpringBoot的主程序类
return builder.sources(DemoApplication.class);
}
}
答案:

@HandlesTypes({WebApplicationInitializer.class})
public class SpringServletContainerInitializer implements ServletContainerInitializer {
public SpringServletContainerInitializer() {
}
public void onStartup(@Nullable Set<Class<?>> webAppInitializerClasses, ServletContext servletContext) throws ServletException {
List<WebApplicationInitializer> initializers = new LinkedList();
Iterator var4;
if (webAppInitializerClasses != null) {
var4 = webAppInitializerClasses.iterator();
while(var4.hasNext()) {
Class<?> waiClass = (Class)var4.next();
if (!waiClass.isInterface() && !Modifier.isAbstract(waiClass.getModifiers()) && WebApplicationInitializer.class.isAssignableFrom(waiClass)) {
try {
// 将@HandlesTypes(WebApplicationInitializer.class)标注的所有这个类型的类都传入到onStartup方法的Set>;为这些WebApplicationInitializer类型的类创建实例。
initializers.add((WebApplicationInitializer)ReflectionUtils.accessibleConstructor(waiClass, new Class[0]).newInstance());
} catch (Throwable var7) {
throw new ServletException("Failed to instantiate WebApplicationInitializer class", var7);
}
}
}
}
if (initializers.isEmpty()) {
servletContext.log("No Spring WebApplicationInitializer types detected on classpath");
} else {
servletContext.log(initializers.size() + " Spring WebApplicationInitializers detected on classpath");
AnnotationAwareOrderComparator.sort(initializers);
var4 = initializers.iterator();
while(var4.hasNext()) {
WebApplicationInitializer initializer = (WebApplicationInitializer)var4.next();
// 每一个WebApplicationInitializer都调用自己的onStartup()
initializer.onStartup(servletContext);
}
}
}
}

public abstract class SpringBootServletInitializer implements WebApplicationInitializer {
public void onStartup(ServletContext servletContext) throws ServletException {
this.logger = LogFactory.getLog(this.getClass());
// 创建web应用容器
WebApplicationContext rootAppContext = this.createRootApplicationContext(servletContext);
if (rootAppContext != null) {
servletContext.addListener(new ContextLoaderListener(rootAppContext) {
public void contextInitialized(ServletContextEvent event) {
}
});
} else {
this.logger.debug("No ContextLoaderListener registered, as createRootApplicationContext() did not return an application context");
}
}
protected WebApplicationContext createRootApplicationContext(ServletContext servletContext) {
// 1、创建SpringApplicationBuilder
SpringApplicationBuilder builder = this.createSpringApplicationBuilder();
StandardServletEnvironment environment = new StandardServletEnvironment();
// 2、准备环境
environment.initPropertySources(servletContext, (ServletConfig)null);
builder.environment(environment);
builder.main(this.getClass());
ApplicationContext parent = this.getExistingRootWebApplicationContext(servletContext);
if (parent != null) {
this.logger.info("Root context already created (using as parent).");
servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE, (Object)null);
builder.initializers(new ApplicationContextInitializer[]{new ParentContextApplicationContextInitializer(parent)});
}
// 3、初始化
builder.initializers(new ApplicationContextInitializer[]{new ServletContextApplicationContextInitializer(servletContext)});
builder.contextClass(AnnotationConfigServletWebServerApplicationContext.class);
// 4、调用configure方法,子类重写了这个方法,将SpringBoot的主程序类传入了进来
builder = this.configure(builder);
SpringApplication application = builder.build();
if (application.getAllSources().isEmpty() && AnnotationUtils.findAnnotation(this.getClass(), Configuration.class) != null) {
application.addPrimarySources(Collections.singleton(this.getClass()));
}
Assert.state(!application.getAllSources().isEmpty(), "No SpringApplication sources have been defined. Either override the configure method or add an @Configuration annotation");
if (this.registerErrorPageFilter) {
application.addPrimarySources(Collections.singleton(ErrorPageFilterConfiguration.class));
}
// 5、启动Spring应用
return this.run(application);
}
}
// 此方法被启动引导类 ServletInitializer有方法重写, 传入的是应用构建器SpringApplicationBuilder, 也就是SpringBoot的主程序类
protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
return builder;
}
https://www.cnblogs.com/hello-shf/p/10864977.html
答案:
通过事件监听的方式读取配置文件,同一个配置,优先级高的覆盖优先级低的,不同配置则取并集。
