哈喽,大家好,我是了不起。
现在Java Web 开发应该都是使用的 SpringBoot,部署的时候直接打包成jar包运行即可。
但是之前用SSH或SSM开发的时候,通常是打包成war包,然后部署到类似Tomcat的web服务器运行。
那么问题来了:为什么 SpringBoot 的 jar 包可以直接运行呢?
1、Fat jar
和普通的 jar 包结构不同,通过 SpringBoot 打包而成的 jar 包是 Fat jar(胖 JAR),在 BOOT-INF/lib 目录下,包含了项目依赖的全部jar 包。在 BOOT-INF/classes 目录下,包含了项目运行的class文件。
这意味着这个 JAR 文件不仅包含了应用的代码,还包括了所有必要的依赖库。这样做的好处就是,我们不需要在运行环境中单独安装这些库了,因为它们已经包含在 JAR 文件中了。
图片
可能会有人问,是如何打包成这种方式的呢?
通常在我们的构建文件中,比如pom.xml 文件:
- <project>
- <!-- ... 其他配置 ... -->
- <build>
- <plugins>
- <plugin>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-maven-plugin</artifactId>
- </plugin>
- </plugins>
- </build>
- </project>
当我们执行 mvn package 命令时,它会创建一个包含所有依赖的 JAR 文件。
2、内嵌服务器
Spring Boot 应用通常内嵌了一个服务器(如 Tomcat)。这是通过在 pom.xml 或 build.gradle 文件中添加相应的依赖实现的。
例如,以下是一个包含 Spring Boot 与 Tomcat 依赖的 pom.xml 片段:
- <dependencies>
- <dependency>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-web</artifactId>
- </dependency>
- <!-- ... 其他依赖 ... -->
- </dependencies>
当 Spring Boot 应用启动时,它会自动配置并启动这个内嵌的 Tomcat 服务器。
3、Spring Boot 启动器
每个 Spring Boot 应用都有一个入口类,这个类包含了 main 方法,它是整个应用启动的起点。这个类使用 @SpringBootApplication 注解,这个注解是一个方便的注解,集成了 @Configuration、@EnableAutoConfiguration 和 @ComponentScan。例如:
import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; @SpringBootApplication public class MyApplication { public static void main(String[] args) { SpringApplication.run(MyApplication.class, args); } }
当运行这个应用时,SpringApplication.run() 方法会启动 Spring 应用上下文,并且启动内嵌的服务器。
4、可执行 JAR 文件
在构建的 JAR 文件中,META-INF/MANIFEST.MF 文件指定了主类(Main-Class)。
当我们使用 java -jar 命令运行 JAR 文件时,Java 虚拟机就知道从哪个类的 main 方法开始执行。
这里的Main-Class是Spring Boot的JarLauncher,它是负责启动整个Spring Boot应用程序的类。
大家可以追踪一下源码:
protected void launch(String[] args) throws Exception { JarFile.registerUrlProtocolHandler(); //自定义类加载器加载jar文件 ClassLoader classLoader = createClassLoader(getClassPathArchives()); //关注getMainClass方法 launch(args, getMainClass(), classLoader); }
@Override protected String getMainClass() throws Exception { Manifest manifest = this.archive.getManifest(); String mainClass = null; if (manifest != null) { mainClass = manifest.getMainAttributes().getValue("Start-Class"); } if (mainClass == null) { throw new IllegalStateException("No 'Start-Class' manifest entry specified in " + this); } return mainClass; }
原文地址:https://mp.weixin.qq.com/s/H3o_z-Qz9FpAsf6rFJjQWA