文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

SpringBoot为什么可以使用Jar包启动

2023-06-29 15:55

关注

这篇文章将为大家详细讲解有关SpringBoot为什么可以使用Jar包启动,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。

引言

很多初学者会比较困惑,Spring Boot 是如何做到将应用代码和所有的依赖打包成一个独立的 Jar 包,因为传统的 Java 项目打包成 Jar 包之后,需要通过 -classpath 属性来指定依赖,才能够运行。

Spring Boot 打包插件

Spring Boot 提供了一个名叫 spring-boot-maven-plugin 的 maven 项目打包插件,如下:

<plugin>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-maven-plugin</artifactId></plugin>

可以方便的将 Spring Boot 项目打成 jar 包。 这样我们就不再需要部署 Tomcat 、Jetty等之类的 Web 服务器容器啦。

我们先看一下 Spring Boot 打包后的结构是什么样的,打开 target 目录我们发现有两个jar包:

SpringBoot为什么可以使用Jar包启动

其中,springboot-0.0.1-SNAPSHOT.jar 是通过 Spring Boot 提供的打包插件采用新的格式打成 Fat Jar,包含了所有的依赖;

springboot-0.0.1-SNAPSHOT.jar.original 则是Java原生的打包方式生成的,仅仅只包含了项目本身的内容。

SpringBoot FatJar 的组织结构

我们将 Spring Boot 打的可执行 Jar 展开后的结构如下所示:

SpringBoot为什么可以使用Jar包启动

我们看到,如果去掉BOOT-INF目录,这将是一个非常普通且标准的Jar包,包括元信息以及可执行的代码部分,其/META-INF/MAINFEST.MF指定了Jar包的启动元信息,org.springframework.boot.loader 执行对应的逻辑操作。

MAINFEST.MF 元信息

元信息内容如下所示:

Manifest-Version: 1.0Spring-Boot-Classpath-Index: BOOT-INF/classpath.idxImplementation-Title: springbootImplementation-Version: 0.0.1-SNAPSHOTSpring-Boot-Layers-Index: BOOT-INF/layers.idxStart-Class: com.listenvision.SpringbootApplicationSpring-Boot-Classes: BOOT-INF/classes/Spring-Boot-Lib: BOOT-INF/lib/Build-Jdk-Spec: 1.8Spring-Boot-Version: 2.5.6Created-By: Maven Jar Plugin 3.2.0Main-Class: org.springframework.boot.loader.JarLauncher

它相当于一个 Properties 配置文件,每一行都是一个配置项目。重点来看看两个配置项:

启动原理

Spring Boot 的启动原理如下图所示:

SpringBoot为什么可以使用Jar包启动

源码分析

JarLauncher

JarLauncher 类是针对 Spring Boot jar 包的启动类, 完整的类图如下所示:

SpringBoot为什么可以使用Jar包启动

其中的 WarLauncher 类,是针对 Spring Boot war 包的启动类。 启动类 org.springframework.boot.loader.JarLauncher 并非为项目中引入类,而是 spring-boot-maven-plugin 插件 repackage 追加进去的。

接下来我们先来看一下 JarLauncher 的源码,比较简单,如下图所示:

public class JarLauncher extends ExecutableArchiveLauncher {    private static final String DEFAULT_CLASSPATH_INDEX_LOCATION = "BOOT-INF/classpath.idx";    static final EntryFilter NESTED_ARCHIVE_ENTRY_FILTER = (entry) -> {        if (entry.isDirectory()) {            return entry.getName().equals("BOOT-INF/classes/");        }        return entry.getName().startsWith("BOOT-INF/lib/");    };        public JarLauncher() {    }        protected JarLauncher(Archive archive) {        super(archive);    }        @Override    protected ClassPathIndexFile getClassPathIndex(Archive archive) throws IOException {        // Only needed for exploded archives, regular ones already have a defined order        if (archive instanceof ExplodedArchive) {            String location = getClassPathIndexFileLocation(archive);            return ClassPathIndexFile.loadIfPossible(archive.getUrl(), location);        }        return super.getClassPathIndex(archive);    }           private String getClassPathIndexFileLocation(Archive archive) throws IOException {        Manifest manifest = archive.getManifest();        Attributes attributes = (manifest != null) ? manifest.getMainAttributes() : null;        String location = (attributes != null) ? attributes.getValue(BOOT_CLASSPATH_INDEX_ATTRIBUTE) : null;        return (location != null) ? location : DEFAULT_CLASSPATH_INDEX_LOCATION;    }        @Override    protected boolean isPostProcessingClassPathArchives() {        return false;    }        @Override    protected boolean isSearchCandidate(Archive.Entry entry) {        return entry.getName().startsWith("BOOT-INF/");    }        @Override    protected boolean isNestedArchive(Archive.Entry entry) {        return NESTED_ARCHIVE_ENTRY_FILTER.matches(entry);    }        public static void main(String[] args) throws Exception {        //调用基类 Launcher 定义的 launch 方法        new JarLauncher().launch(args);    }}

主要看它的 main 方法,调用的是基类 Launcher 定义的 launch 方法,而 Launcher 是ExecutableArchiveLauncher 的父类。下面我们来看看Launcher基类源码:

Launcher

public abstract class Launcher {    private static final String JAR_MODE_LAUNCHER = "org.springframework.boot.loader.jarmode.JarModeLauncher";    protected void launch(String[] args) throws Exception {        if (!isExploded()) {            JarFile.registerUrlProtocolHandler();        }        ClassLoader classLoader = createClassLoader(getClassPathArchivesIterator());        String jarMode = System.getProperty("jarmode");        String launchClass = (jarMode != null && !jarMode.isEmpty()) ? JAR_MODE_LAUNCHER : getMainClass();        launch(args, launchClass, classLoader);    }        @Deprecated    protected ClassLoader createClassLoader(List<Archive> archives) throws Exception {        return createClassLoader(archives.iterator());    }        protected ClassLoader createClassLoader(Iterator<Archive> archives) throws Exception {        List<URL> urls = new ArrayList<>(50);        while (archives.hasNext()) {            urls.add(archives.next().getUrl());        }        return createClassLoader(urls.toArray(new URL[0]));    }        protected ClassLoader createClassLoader(URL[] urls) throws Exception {        return new LaunchedURLClassLoader(isExploded(), getArchive(), urls, getClass().getClassLoader());    }        protected void launch(String[] args, String launchClass, ClassLoader classLoader) throws Exception {        Thread.currentThread().setContextClassLoader(classLoader);        createMainMethodRunner(launchClass, args, classLoader).run();    }        protected MainMethodRunner createMainMethodRunner(String mainClass, String[] args, ClassLoader classLoader) {        return new MainMethodRunner(mainClass, args);    }    protected abstract String getMainClass() throws Exception;        protected Iterator<Archive> getClassPathArchivesIterator() throws Exception {        return getClassPathArchives().iterator();    }        @Deprecated    protected List<Archive> getClassPathArchives() throws Exception {        throw new IllegalStateException("Unexpected call to getClassPathArchives()");    }        protected final Archive createArchive() throws Exception {        ProtectionDomain protectionDomain = getClass().getProtectionDomain();        CodeSource codeSource = protectionDomain.getCodeSource();        URI location = (codeSource != null) ? codeSource.getLocation().toURI() : null;        String path = (location != null) ? location.getSchemeSpecificPart() : null;        if (path == null) {            throw new IllegalStateException("Unable to determine code source archive");        }        File root = new File(path);        if (!root.exists()) {            throw new IllegalStateException("Unable to determine code source archive from " + root);        }        return (root.isDirectory() ? new ExplodedArchive(root) : new JarFileArchive(root));    }        protected boolean isExploded() {        return false;    }        protected Archive getArchive() {        return null;    }}

PropertiesLauncher

@Overrideprotected String getMainClass() throws Exception {    //加载 jar包 target目录下的  MANIFEST.MF 文件中 Start-Class配置,找到springboot的启动类    String mainClass = getProperty(MAIN, "Start-Class");    if (mainClass == null) {        throw new IllegalStateException("No '" + MAIN + "' or 'Start-Class' specified");    }    return mainClass;}

MainMethodRunner

目标类main方法的执行器,此时的 mainClassName 被赋值为 MANIFEST.MF 中配置的 Start-Class 属性值,也就是 com.listenvision.SpringbootApplication,之后便是通过反射执行 SpringbootApplication 的 main 方法,从而达到启动 Spring Boot 的效果。

public class MainMethodRunner {    private final String mainClassName;    private final String[] args;    public MainMethodRunner(String mainClass, String[] args) {        this.mainClassName = mainClass;        this.args = (args != null) ? args.clone() : null;    }    public void run() throws Exception {        Class<?> mainClass = Class.forName(this.mainClassName, false, Thread.currentThread().getContextClassLoader());        Method mainMethod = mainClass.getDeclaredMethod("main", String[].class);        mainMethod.setAccessible(true);        mainMethod.invoke(null, new Object[] { this.args });    }}

关于“SpringBoot为什么可以使用Jar包启动”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯