这篇文章主要介绍了三分钟带你了解SpringBoot真正的启动引导类,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
引言
SpringBoot项目中的启动类,一般都是XXApplication,例如**「StatsApplication」,「UnionApplication」**。
每个项目的启动类名称都不一样。但是它的启动类真的是XXApplication吗?
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,竟然找不到!!
进入 https://search.maven.org/classic/#advancedsearch 查询JarLauncher
在查询结果找到spring下的项目
确定 JarLauncher 位于 spring-boot-loader 下。为了方便查看源码,在 pom 中引入
org.springframework.bootspring-boot-loaderprovided
3.2.JarLauncher说明
JarLauncher作为引导类 ,当调用java -jar 命令时,将调用 main 方法,实际上调用的是 **「JarLauncher#launch」**方法,该方法继承于 org.springframework.boot.loader.Launcher
简化层次关系为:
JarLauncher#launch代码如下
protected void launch(String[] args) throws Exception { JarFile.registerUrlProtocolHandler(); ClassLoader classLoader = createClassLoader(getClasspathArchives()); launch(args, getMainClass(), classLoader); }
「聚句解析」
「1,.JarFile.registerUrlProtocolHandler();」
Spring Boot 生成的 FAT jar,在被 java -jar 引导时,其内部的 jar 文件无法被 sun.net.www.protocol.jar.Handler 处理。微信公众号搜索, [Java学习之道] ,回复 ‘福利' 2T 资料等你来拿~
所以 SpringBoot 实现了,org.springframework.boot.loader.jar.Handler
JarFile.registerUrlProtocolHandler(), 就注册 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()) { return entry.getName().equals(BOOT_INF_CLASSES); } return entry.getName().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 { Class> mainClass = Thread.currentThread().getContextClassLoader() .loadClass(this.mainClassName); Method mainMethod = mainClass.getDeclaredMethod("main", String[].class); 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中。」
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。