引言

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.JarLauncherStart-Class: com.shanyuan.StatsApplication

也就是说:「org.springframework.boot.loader.JarLauncher 是 SpringBoot 的启动类!」

上面浏览下 JarLauncher

浏览JarLauncher

3.1 找到JarLauncher

进入IDEA,Ctrl+N查找JarLauncher,居然找不到!!

进入 https://search.maven.org/clas... 查问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 {   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/";@Overrideprotected 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中。」

写在最初

欢送大家关注我的公众号【惊涛骇浪如码】,海量Java相干文章,学习材料都会在外面更新,整顿的材料也会放在外面。

感觉写的还不错的就点个赞,加个关注呗!点关注,不迷路,继续更新!!!