三分钟带你了解SpringBoot真正的启动引导类引⾔
SpringBoot项⽬中的启动类,⼀般都是XXApplication,例如**「StatsApplication」,「UnionApplication」**。
每个项⽬的启动类名称都不⼀样。但是它的启动类真的是XXApplication吗?
**META-INF/**Manifest.mf⽂件
jar⽂件实际上是class⽂件的zip压缩存档。jar并不能表达应⽤程序的便签信息.
「META-INF/Manifest.mf⽂件提供存档的便签信息.」
Manifest.mf有「Main-Class,⽤来标明jar⽂件的⼊⼝类。」
解压jar包,查看META-INF/Manifest.mf过程如下:
重要信息如下
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.shanyuan.StatsApplication
也就是说:「org.springframework.boot.loader.JarLauncher 是 SpringBoot 的启动类!」
下⾯浏览下JarLauncher
浏览JarLauncher
3.1 到JarLauncher
进⼊IDEA,Ctrl+N查JarLauncher,竟然不到!!
在查询结果到spring下的项⽬
确定JarLauncher 位于 spring-boot-loader下。为了⽅便查看源码,在 pom 中引⼊
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-loader</artifactId>
<scope>provided</scope>
</dependency>
3.2.JarLauncher说明
JarLauncher作为引导类,当调⽤java -jar命令时,将调⽤ main ⽅法,实际上调⽤的是 **「JarLauncher#launch」**⽅法,该⽅法继承
于org.springframework.boot.loader.Launcher
简化层次关系为:
JarLauncher#launch代码如下
protected void launch(String[] args) throws Exception {
ClassLoader classLoader = createClassLoader(getClassPathArchives());
launch(args, getMainClass(), classLoader);
}
「聚句解析」
「1,.isterUrlProtocolHandler();」
Spring Boot ⽣成的 FAT jar,在被 java -jar 引导时,其内部的 jar ⽂件⽆法被sun.www.protocol.jar.Handler处理。搜索, [Java学习之道] ,回复‘福利' 2T 资料等你来拿~
所以 SpringBoot 实现了,org.springframework.boot.loader.jar.Handler
「2.ClassLoader classLoader = createClassLoader(getClassPathArchives());」
创建ClassLoader。
getClassPathArchives核⼼判断是isNestedArchive⽅法。
isNestedArchive 被 JarLauncher 覆写了。其实现如下:
static final String BOOT_INF_CLASSES = "BOOT-INF/classes/";
static final String BOOT_INF_LIB = "BOOT-INF/lib/";
@Override
protected boolean isNestedArchive(Archive.Entry entry) {
if (entry.isDirectory()) {
Name().equals(BOOT_INF_CLASSES);
}
Name().startsWith(BOOT_INF_LIB);
}
也就是说,只要 **「满⾜以BOOT-INF/classes/和BOOT-INF/lib/都是classLoader加载」**的范围。
解压的jar,查看也与只对应
3. launch(args, getMainClass(), classLoader);
protected void launch(String[] args, String mainClass,
ClassLoader classLoader)
throws Exception {
Thread.currentThread().setContextClassLoader(classLoader);
createMainMethodRunner(mainClass, args, classLoader).run();
}
查看 createMainMethodRunner 的 run ⽅法,如下:
public class MainMethodRunner {
// 省略部分代码
public void run() throws Exception {
springframework和springboot
Class<?> mainClass = Thread.currentThread().getContextClassLoader()
.loadClass(this.mainClassName);
Method mainMethod =
mainMethod.invoke(null, new Object[] { this.args });
}
}
其中 mainClass,来⾃/META-INF/MANIFEST.MF中的Start-Class属性。
「即,JarLauncher 是同进程内,通过反射调⽤ Start-Class 对应类,即 XXXApplication 的 main ⽅法。」
4.总结
SpringBoot 项⽬的实际启动类是org.springframework.boot.loader.JarLauncher。
「在 JarLauncher 内部通过反射调⽤ XXApplication 类的 main ⽅法。具体实现位于 MainMethodRunner中。」
到此这篇关于三分钟带你了解SpringBoot真正的启动引导类的⽂章就介绍到这了,更多相关SpringBoot 启动引导类内容请搜索以前的⽂章或继续浏览下⾯的相关⽂章希望⼤家以后多多⽀持!

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。