三分钟内如何识别SpringBoot的核心启动引导类?

2026-05-16 00:361阅读0评论SEO资讯
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计1199个文字,预计阅读时间需要5分钟。

三分钟内如何识别SpringBoot的核心启动引导类?

SpringBoot项目中,启动类通常都是XXApplication,例如StatsApplication、UnionApplication等。每个项目的启动类名称都不一样。但它的启动类真的是XXApplication吗?

在Spring Boot项目中,启动类通常以`Application`结尾,如`StatsApplication`或`UnionApplication`。这些类负责启动Spring应用程序,并配置Spring上下文。启动类通常包含`@SpringBootApplication`注解,这个注解是Spring Boot的核心,它告诉Spring Boot这是一个Spring应用程序的入口点。

`@SpringBootApplication`注解本身不指定具体的启动类,而是通过Spring的类扫描机制自动找到标注了该注解的类。因此,尽管每个项目的启动类名称可能不同,但它们的启动类本质上都是继承自`SpringApplication.run()`方法。

`META-INF/MANIFEST.MF`文件中的信息通常与Java应用程序的打包和部署有关,它不直接关联到启动类。这个文件包含了一些元数据,如主类信息,但Spring Boot的启动过程不会直接从`META-INF/MANIFEST.MF`中读取启动类信息。

总结来说,尽管每个项目的启动类名称可能不同,但它们都是通过`@SpringBootApplication`注解启动的,并且本质上都是通过`SpringApplication.run()`方法来启动Spring应用程序的。

引言

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,竟然找不到!!

进入 search.maven.org/classic/#advancedsearch 查询JarLauncher

三分钟内如何识别SpringBoot的核心启动引导类?

在查询结果找到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/"; @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中。」

到此这篇关于三分钟带你了解SpringBoot真正的启动引导类的文章就介绍到这了,更多相关SpringBoot 启动引导类内容请搜索易盾网络以前的文章或继续浏览下面的相关文章希望大家以后多多支持易盾网络!

标签:启动

本文共计1199个文字,预计阅读时间需要5分钟。

三分钟内如何识别SpringBoot的核心启动引导类?

SpringBoot项目中,启动类通常都是XXApplication,例如StatsApplication、UnionApplication等。每个项目的启动类名称都不一样。但它的启动类真的是XXApplication吗?

在Spring Boot项目中,启动类通常以`Application`结尾,如`StatsApplication`或`UnionApplication`。这些类负责启动Spring应用程序,并配置Spring上下文。启动类通常包含`@SpringBootApplication`注解,这个注解是Spring Boot的核心,它告诉Spring Boot这是一个Spring应用程序的入口点。

`@SpringBootApplication`注解本身不指定具体的启动类,而是通过Spring的类扫描机制自动找到标注了该注解的类。因此,尽管每个项目的启动类名称可能不同,但它们的启动类本质上都是继承自`SpringApplication.run()`方法。

`META-INF/MANIFEST.MF`文件中的信息通常与Java应用程序的打包和部署有关,它不直接关联到启动类。这个文件包含了一些元数据,如主类信息,但Spring Boot的启动过程不会直接从`META-INF/MANIFEST.MF`中读取启动类信息。

总结来说,尽管每个项目的启动类名称可能不同,但它们都是通过`@SpringBootApplication`注解启动的,并且本质上都是通过`SpringApplication.run()`方法来启动Spring应用程序的。

引言

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,竟然找不到!!

进入 search.maven.org/classic/#advancedsearch 查询JarLauncher

三分钟内如何识别SpringBoot的核心启动引导类?

在查询结果找到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/"; @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中。」

到此这篇关于三分钟带你了解SpringBoot真正的启动引导类的文章就介绍到这了,更多相关SpringBoot 启动引导类内容请搜索易盾网络以前的文章或继续浏览下面的相关文章希望大家以后多多支持易盾网络!

标签:启动