SpringBoot的jar可以直接运行的原因主要包括Maven或Gradle插件的支持、Fat JAR的结构、MANIFEST.MF文件的作用、JarLauncher的启动机制、嵌入式Servlet容器的使用以及自动化配置的特性。面试遇到这个问题时,可从以下几点进行回答:
1. Maven或Gradle插件的支持
SpringBoot项目通常使用Maven或Gradle作为构建工具,并通过相应的插件(如Maven的spring-boot-maven-plugin或Gradle的spring-boot-gradle-plugin)来支持打包操作。这些插件在打包过程中会执行一系列任务,将应用程序及其所有依赖项打包成一个可执行的jar文件。这个jar文件包含了运行应用程序所需的所有资源,使得它可以在没有安装额外依赖的情况下直接运行。
2. Fat JAR(可执行JAR)的结构
SpringBoot的jar文件通常被称为Fat JAR或Uber JAR,因为它包含了比普通jar文件更多的内容。具体来说,Fat JAR的结构通常包括以下几个部分:
META-INF:包含jar包的元数据信息,如MANIFEST.MF文件。这个文件是jar包的重要组成部分,它指定了jar包的入口点(即主类)和其他相关信息。
BOOT-INF:这是Fat JAR特有的部分,它包含了应用程序的编译代码(classes目录)和所有依赖的jar包(lib目录)。这些依赖的jar包被打包在Fat JAR内部,以便在运行时能够直接加载。
org/springframework/boot/loader:这个目录包含了SpringBoot的加载器相关类,如JarLauncher。这些类负责在jar包运行时加载应用程序的Main方法和其他资源。
3. MANIFEST.MF文件的作用
MANIFEST.MF文件是jar包的清单文件,它位于META-INF目录下。这个文件包含了关于jar包的元数据信息,其中最重要的是Main-Class属性。在SpringBoot的Fat JAR中,Main-Class属性通常被设置为org.springframework.boot.loader.JarLauncher。这意味着当执行java -jar命令时,JVM会加载JarLauncher类,并调用其main方法来启动应用程序。
4. JarLauncher的启动机制
JarLauncher是SpringBoot jar包启动的入口点。它的主要作用是创建一个自定义的类加载器(如LaunchedURLClassLoader),用于加载BOOT-INF/lib目录下的jar文件(包括Spring Boot loader相关的类和应用程序的依赖库)。然后,它会在一个新线程中启动应用程序的Main方法(即Start-Class在MANIFEST.MF中指定的类),从而启动应用程序。
5. 嵌入式Servlet容器的使用
SpringBoot默认使用嵌入式的Servlet容器(如Tomcat、Jetty、Undertow等),这使得应用程序可以作为一个独立的web应用运行,而无需部署到外部容器中。在打包成jar文件时,这些嵌入式Servlet容器也被包含在内,使得jar文件可以直接作为web应用启动。这种设计大大简化了应用程序的部署和启动过程。
6. 自动化配置
SpringBoot还提供了自动化配置的特性,它根据应用程序的依赖和配置信息,自动对应用程序进行一些常见的配置。这种自动化配置减少了开发者在配置方面的工作量,使得应用程序更容易打包成一个独立的可执行jar文件。例如,当检测到Spring Web Starter依赖时,SpringBoot会自动配置嵌入式Servlet容器和相关的Web组件。