面试官:为什么SpringBoot的 jar 可以直接运行?

开发 前端
和普通的 jar 包结构不同,通过 SpringBoot 打包而成的 jar 包是 Fat jar(胖 JAR),在 BOOT-INF/lib 目录下,包含了项目依赖的全部jar 包。在 BOOT-INF/classes 目录下,包含了项目运行的class文件。

哈喽,大家好,我是了不起。

现在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;
}
责任编辑:武晓燕 来源: Java技术指北
相关推荐

2022-02-13 20:31:26

SpringBootjarmanifests

2024-04-03 09:01:34

SpringTomcat容器

2020-05-07 16:30:32

Spring BootJava

2021-02-19 10:02:57

HTTPSJava安全

2022-07-06 13:48:24

RedisSentinel机制

2021-01-21 07:53:29

面试官Promis打印e

2023-12-06 09:10:28

JWT微服务

2020-10-24 15:50:54

Java值传递代码

2021-12-28 09:50:18

Redis单线程高并发

2022-12-27 08:39:54

MySQL主键索引

2021-12-20 10:30:33

forforEach前端

2024-03-18 14:06:00

停机Spring服务器

2023-12-20 14:35:37

Java虚拟线程

2023-06-05 07:57:53

Kafka消息事务消息

2022-12-22 14:32:37

JavaScript编程语言

2023-07-05 08:17:38

JDK动态代理接口

2024-03-13 07:53:57

弱引用线程工具

2024-01-19 14:03:59

Redis缓存系统Spring

2024-11-21 10:38:10

2021-09-07 10:44:33

Java 注解开发
点赞
收藏

51CTO技术栈公众号