Spring Boot 启动类真的是XXApplication?

    科技2025-03-28  11

    1. 引言

    SpringBoot项目中的启动类,一般都是XXApplication,例如StatsApplication,UnionApplication。

    每个项目的启动类名称都不一样。

    但是它的启动类真的是XXApplication吗?

    2. 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 才是 Spring Boot 的启动类!

    不熟悉 Spring Boot 的可以看下这个仓库:https://github.com/javastacks/spring-boot-best-practice

    下面浏览下JarLauncher

    3. 浏览JarLauncher

    3.1 找到JarLauncher

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

    进入 https://search.maven.org/classic/#advancedsearch 查询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处理。

    所以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.总结

    Spring Boot 项目的实际启动类是org.springframework.boot.loader.JarLauncher。

    在JarLauncher内部通过反射调用XXApplication类的main方法。

    具体实现位于 MainMethodRunner中。最后,关于 Spring Boot 系列教程可以关注公众号互联网架构师获取阅读。

    作者:温安适来源:https://my.oschina.net/floor/blog/4301613

    关微信公众号:互联网架构师,在后台回复:2T,可以获取我整理的教程,都是干货。

    猜你喜欢

    1、GitHub 标星 3.2w!史上最全技术人员面试手册!FackBook发起和总结

    2、如何才能成为优秀的架构师?

    3、从零开始搭建创业公司后台技术栈

    4、程序员一般可以从什么平台接私活?

    5、37岁程序员被裁,120天没找到工作,无奈去小公司,结果懵了...

    6、滴滴业务中台构建实践,首次曝光

    7、不认命,从10年流水线工人,到谷歌上班的程序媛,一位湖南妹子的励志故事

    8、15张图看懂瞎忙和高效的区别

    9、2T架构师学习资料干货分享

    Processed: 0.013, SQL: 8