在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Spring Boot的啟動原理

科技綠洲 ? 來源:Java技術指北 ? 作者:Java技術指北 ? 2023-10-13 11:44 ? 次閱讀

可能很多初學者會比較困惑,Spring Boot 是如何做到將應用代碼和所有的依賴打包成一個獨立的 Jar 包,因為傳統的 Java 項目打包成 Jar 包之后,需要通過 -classpath 屬性來指定依賴,才能夠運行。我們今天就來分析講解一下 Spring Boot 的啟動原理。

1. Spring Boot 打包插件

Spring Boot 提供了一個名叫 spring-boot-maven-plugin 的 maven 項目打包插件,可以方便的將 Spring Boot 項目打成 jar 包。這樣我們就不再需要部署 Tomcat 、Jetty等之類的 Web 服務器容器啦。

我們先看一下 Spring Boot 打包后的結構是什么樣的,打開 target 目錄我們發現有兩個jar包:

  1. hello-0.0.1-SNAPSHOT.jar:17.3MB
  2. hello-0.0.1-SNAPSHOT.jar.original:3KB

其中,hello-0.0.1-SNAPSHOT.jar 是通過 Spring Boot 提供的打包插件采用新的格式打成 Fat Jar,包含了所有的依賴;而 hello-0.0.1-SNAPSHOT.jar.original 則是Java原生的打包方式生成的,僅僅只包含了項目本身的內容。

2. SpringBoot FatJar 的組織結構

我們將 Spring Boot 打的可執行 Jar 展開后的結構如下所示:

.
├── BOOT-INF
│   ├── classes
│   │   ├── application.properties
│   │   └── com
│   │       └── javanorth
│   │           └── hello
│   │               └── HelloApplication.class
│   └── lib
│       ├── spring-boot-2.5.0.RELEASE.jar
│       ├── spring-boot-autoconfigure-2.5.0.RELEASE.jar
│       ├── spring-boot-configuration-processor-2.5.0.RELEASE.jar
│       ├── spring-boot-starter-2.5.0.RELEASE.jar
│       ├── ...
├── META-INF
│   ├── MANIFEST.MF
│   └── maven
│       └── com.javanorth
│           └── hello
│               ├── pom.properties
│               └── pom.xml
│   
├── org
│   └── springframework
│       └── boot
│           └── loader
│               ├── ExecutableArchiveLauncher.class
│               ├── JarLauncher.class
│               ├── Launcher.class
│               ├── MainMethodRunner.class
│               ├── ...
  • BOOT-INF目錄:包含了我們的項目代碼(classes目錄),以及所需要的依賴(lib 目錄)
  • META-INF目錄:通過 MANIFEST.MF 文件提供 Jar包的元數據,聲明了 jar 的啟動類
  • org.springframework.boot.loader :Spring Boot 的加載器代碼,實現的 Jar in Jar 加載的魔法源

我們看到,如果去掉BOOT-INF目錄,這將是一個非常普通且標準的Jar包,包括元信息以及可執行的代碼部分,其/META-INF/MAINFEST.MF指定了Jar包的啟動元信息,org.springframework.boot.loader 執行對應的邏輯操作。

3. MAINFEST.MF 元信息分析

元信息內容如下所示:

Manifest-Version: 1.0
Created-By: Maven Jar Plugin 3.2.0
Build-Jdk-Spec: 11
Implementation-Title: hello
Implementation-Version: 0.0.1-SNAPSHOT
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.javanorth.hello.HelloApplication
Spring-Boot-Version: 2.5.0
Spring-Boot-Classes: BOOT-INF/classes/
Spring-Boot-Lib: BOOT-INF/lib/
Spring-Boot-Classpath-Index: BOOT-INF/classpath.idx
Spring-Boot-Layers-Index: BOOT-INF/layers.idx

它相當于一個 Properties 配置文件,每一行都是一個配置項目。重點來看看兩個配置項:

  • Main-Class 配置項:Java 規定的 jar 包的啟動類,這里設置為 spring-boot-loader 項目的 JarLauncher 類,進行 Spring Boot 應用的啟動。
  • Start-Class 配置項:Spring Boot 規定的主啟動類,這里設置為我們定義的 Application 類。
  • Spring-Boot-Classes 配置項:指定加載應用類的入口
  • Spring-Boot-Lib 配置項: 指定加載應用依賴的庫

4. 啟動原理

Spring Boot 的啟動原理如下圖所示:

圖片

5. 源碼分析

5.1 org.springframework.boot.loader.JarLauncher

JarLauncher 類是針對 Spring Boot jar 包的啟動類, 完整的類圖如下所示:

圖片
Spring Boot Start jar 2

其中的 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 {
        new JarLauncher().launch(args);
    }
}

當執行 java -jar 命令或執行解壓后的 org.springframework.boot.loader.JarLauncher 類時,JarLauncher 會將 BOOT-INF/classes 下的類文件和 BOOT-INF/lib 下依賴的jar加入到classpath下,后調用 META-INF/MANIFEST.MF 文件 Start-Class 屬性 [指向項目中的 com.javanorth.hello.HelloApplicatioin 啟動類] 完成應用程序的啟動。

JarLauncher 假定依賴項jar包含在 /BOOT-INF/lib 目錄中,并且應用程序類包含在 /BOOT-INF/classes 目錄中。它的 main 方法調用的則是基類 Launcher 定義的 launch 方法,而 Launcher 是ExecutableArchiveLauncher 的父類。

5.2 org.springframework.boot.loader.ExecutableArchiveLauncher

ExecutableArchiveLauncher 是 JarLauncher 的直接父類,繼承了 Launcher 基類,并實現部分抽象方法

public abstract class ExecutableArchiveLauncher extends Launcher {
    private static final String START_CLASS_ATTRIBUTE = "Start-Class";
    protected static final String BOOT_CLASSPATH_INDEX_ATTRIBUTE = "Spring-Boot-Classpath-Index";
    private final Archive archive;
    private final ClassPathIndexFile classPathIndex;
    public ExecutableArchiveLauncher() {
        try {
            this.archive = createArchive();
            this.classPathIndex = getClassPathIndex(this.archive);
        }
        catch (Exception ex) {
            throw new IllegalStateException(ex);
        }
    }
    protected ExecutableArchiveLauncher(Archive archive) {
        try {
            this.archive = archive;
            this.classPathIndex = getClassPathIndex(this.archive);
        }
        catch (Exception ex) {
            throw new IllegalStateException(ex);
        }
    }
    protected ClassPathIndexFile getClassPathIndex(Archive archive) throws IOException {
        return null;
    }
    @Override
    protected String getMainClass() throws Exception {
        Manifest manifest = this.archive.getManifest();
        String mainClass = null;
        if (manifest != null) {
            mainClass = manifest.getMainAttributes().getValue(START_CLASS_ATTRIBUTE);
        }
        if (mainClass == null) {
            throw new IllegalStateException("No 'Start-Class' manifest entry specified in " + this);
        }
        return mainClass;
    }
    @Override
    protected ClassLoader createClassLoader(Iterator< Archive > archives) throws Exception {
        List< URL > urls = new ArrayList<  >(guessClassPathSize());
        while (archives.hasNext()) {
            urls.add(archives.next().getUrl());
        }
        if (this.classPathIndex != null) {
            urls.addAll(this.classPathIndex.getUrls());
        }
        return createClassLoader(urls.toArray(new URL[0]));
    }
    private int guessClassPathSize() {
        if (this.classPathIndex != null) {
            return this.classPathIndex.size() + 10;
        }
        return 50;
    }
    @Override
    protected Iterator< Archive > getClassPathArchivesIterator() throws Exception {
        Archive.EntryFilter searchFilter = this::isSearchCandidate;
        Iterator< Archive > archives = this.archive.getNestedArchives(searchFilter,
                (entry) - > isNestedArchive(entry) && !isEntryIndexed(entry));
        if (isPostProcessingClassPathArchives()) {
            archives = applyClassPathArchivePostProcessing(archives);
        }
        return archives;
    }
    private boolean isEntryIndexed(Archive.Entry entry) {
        if (this.classPathIndex != null) {
            return this.classPathIndex.containsEntry(entry.getName());
        }
        return false;
    }
    private Iterator< Archive > applyClassPathArchivePostProcessing(Iterator< Archive > archives) throws Exception {
        List< Archive > list = new ArrayList<  >();
        while (archives.hasNext()) {
            list.add(archives.next());
        }
        postProcessClassPathArchives(list);
        return list.iterator();
    }
    protected boolean isSearchCandidate(Archive.Entry entry) {
        return true;
    }
    protected abstract boolean isNestedArchive(Archive.Entry entry);
    protected boolean isPostProcessingClassPathArchives() {
        return true;
    }
    protected void postProcessClassPathArchives(List< Archive > archives) throws Exception {
    }
    @Override
    protected boolean isExploded() {
        return this.archive.isExploded();
    }
    @Override
    protected final Archive getArchive() {
        return this.archive;
    }
}

5.3 org.springframework.boot.loader.Launcher

如下則是 Launcher 的源碼

  1. launch 方法會首先創建類加載器,而后判斷 jar 是否在 MANIFEST.MF 文件中設置了 jarmode 屬性。
  2. 如果沒有設置,launchClass 的值就來自 getMainClass() 返回,該方法由子類實現,返回 MANIFEST.MF 中配置的 START_CLASS_ATTRIBUTE 屬性值
  3. 調用 createMainMethodRunner 方法,構建一個 MainMethodRunner 對象并調用其 run 方法

jarmode 是創建 docker 鏡像時用到的參數,使用該參數是為了生成帶有多個 layer 信息的鏡像,這里暫不注意

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;
    }
}

5.4 org.springframework.boot.loader.MainMethodRunner

從名字可以判斷這是一個目標類main方法的執行器,此時的 mainClassName 被賦值為 MANIFEST.MF 中配置的 START_CLASS_ATTRIBUTE 屬性值,也就是 com.javanorth.hello.HelloApplication,之后便是通過反射執行 HelloApplication 的 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 });
    }
}

總結

  1. jar 包類似于 zip 壓縮文件,只不過相比 zip 文件多了一個 META-INF/MANIFEST.MF 文件,該文件在構建 jar 包時自動創建
  2. 想要制作可執行 JAR 包,在 MANIFEST.MF 中指定 Main-Class 是關鍵。使用 java 執行 jar 包的時候,實際上等同于使用 java 命令執行指定的 Main-Class 程序。
  3. Spring Boot 提供了一個插件 spring-boot-maven-plugin ,用于把程序打包成一個可執行的jar包
  4. 使用 java -jar 啟動 Spring Boot 的 jar 包,首先調用的入口類是 JarLauncher,內部調用 Launcher 的 launch 后構建 MainMethodRunner 對象,最終通過反射調用 HelloApplication 的 main 方法實現啟動效果。
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 數據
    +關注

    關注

    8

    文章

    7030

    瀏覽量

    89034
  • 服務器
    +關注

    關注

    12

    文章

    9160

    瀏覽量

    85421
  • 代碼
    +關注

    關注

    30

    文章

    4788

    瀏覽量

    68612
  • SpringBoot
    +關注

    關注

    0

    文章

    173

    瀏覽量

    179
收藏 人收藏

    評論

    相關推薦

    啟動Spring Boot項目應用的三種方法

    基礎。我們知道了Spring Boot是個什么了,那么我們又該如何啟動Spring Boot應用呢?這里小編給大家推薦常用的三種方法。分別是
    發表于 01-14 17:33

    Spring Boot嵌入式Web容器原理是什么

    ,不需要配置任何特殊的XML配置,為了這個目標,Spring BootSpring 4.0框架之上提供了很多特性,幫助應用以“約定優于配置”“開箱即用”的方式來啟動應用并運行上下文。
    發表于 12-16 07:57

    Spring Boot定時任務的重寫方法

    Spring Boot應該是目前最火的java開源框架了,它簡化了我們創建一個web服務的過程,讓我們可以在很短時間、基本零配置就可以啟動一個web服務。
    的頭像 發表于 01-20 17:38 ?2449次閱讀

    Spring Boot從零入門1 詳述

    在開始學習Spring Boot之前,我之前從未接觸過Spring相關的項目,Java基礎還是幾年前自學的,現在估計也忘得差不多了吧,寫Spring
    的頭像 發表于 12-10 22:18 ?639次閱讀

    Spring認證」什么是Spring GraphQL?

    這個項目建立在 Boot 2.x 上,但它應該與最新的 Boot2.4.x5 相關。 要創建項目,請轉到start.spring.io并為要使用的GraphQL傳輸選擇啟動器:
    的頭像 發表于 08-10 14:08 ?825次閱讀
    「<b class='flag-5'>Spring</b>認證」什么是<b class='flag-5'>Spring</b> GraphQL?

    Spring Boot特有的實踐

    Spring Boot是最流行的用于開發微服務的Java框架。在本文中,我將與你分享自2016年以來我在專業開發中使用Spring Boot所采用的最佳實踐。這些內容是基于我的個人經驗
    的頭像 發表于 09-29 10:24 ?913次閱讀

    強大的Spring Boot 3.0要來了

    來源:OSC開源社區(ID:oschina2013) Spring Boot 3.0 首個 RC 已發布,此外還為兩個分支發布了更新:2.7.5 2.6.13。 3.0.0-RC1: https
    的頭像 發表于 10-31 11:17 ?1875次閱讀

    Spring Boot啟動優化實踐

    公司 SpringBoot 項目在日常開發過程中發現服務啟動過程異常緩慢,常常需要6-7分鐘才能暴露端口,嚴重降低開發效率。
    的頭像 發表于 02-23 10:26 ?576次閱讀

    Spring Boot Web相關的基礎知識

    上一篇文章我們已經學會了如何通過IDEA快速建立一個Spring Boot項目,還介紹了Spring Boot項目的結構,介紹了項目配置文件pom.xml的組成部分,并且撰寫了我們
    的頭像 發表于 03-17 15:03 ?659次閱讀

    SpringBoot的嵌入式Web容器是什么時候加載的?

    背景:最近有位開發同學說面試被問到Spring Boot啟動流程,以及被問到Spring Boot 的嵌入式Web容器是什么時候加載的。
    的頭像 發表于 07-11 10:10 ?514次閱讀
    SpringBoot的嵌入式Web容器是什么時候加載的?

    Spring Boot配置加載相關知識

    Spring BOOT 啟動參數 在Java Web的開發完成后,以前我們都會打包成war文件,然后放大web容器,比如tomcat、jetty這樣的容器。現在基于SpringBoot開發的項目
    的頭像 發表于 10-07 15:47 ?490次閱讀

    Spring Boot Actuator快速入門

    不知道大家在寫 Spring Boot 項目的過程中,使用過 Spring Boot Actuator 嗎?知道 Spring
    的頭像 發表于 10-09 17:11 ?639次閱讀

    Spring Boot啟動 Eureka流程

    在上篇中已經說過了 Eureka-Server 本質上是一個 web 應用的項目,今天就來看看 Spring Boot 是怎么啟動 Eureka 的。 Spring
    的頭像 發表于 10-10 11:40 ?894次閱讀
    <b class='flag-5'>Spring</b> <b class='flag-5'>Boot</b><b class='flag-5'>啟動</b> Eureka流程

    Spring Boot 的設計目標

    什么是Spring Boot Spring BootSpring 開源組織下的一個子項目,也是 S
    的頭像 發表于 10-13 14:56 ?587次閱讀
    <b class='flag-5'>Spring</b> <b class='flag-5'>Boot</b> 的設計目標

    springboot啟動流程

    Spring Boot啟動流程可以分為以下幾個步驟:初始化啟動環境、加載自動配置類、創建 Spring 上下文、
    的頭像 發表于 11-22 16:04 ?655次閱讀
    主站蜘蛛池模板: 亚洲小视频| 国产精品视频永久免费播放| 天堂中文在线网| 亚洲欧美一区二区三区麻豆| 一级毛毛片毛片毛片毛片在线看| 激情五月激情综合色区| 狠狠狠狠干| 在线 你懂的| 五月婷婷六月合| 色婷婷激婷婷深爱五月小说| 青草悠悠视频在线观看| 欧美成人午夜视频| 精品国产欧美一区二区最新| 国产精品亚洲色图| 222.www免费观看| 手机福利视频| www天天操| 香蕉成人国产精品免费看网站 | 欧美爱爱网址| 精品伊人久久香线蕉| 国产码一区二区三区| 亚洲va国产日韩欧美精品色婷婷| 天堂中文最新版www| 国产在线精品香蕉综合网一区 | 一区二区中文字幕在线观看| 美女张开腿露出尿口让男人桶| 亚洲色吧| 欧美一区二区三区四区在线观看| 九九99视频在线观看视频观看| bt天堂网www连接| 濑亚美莉vs黑人欧美视频| 欧美熟色妇| 超h 高h 污肉1v1御书屋| 偷偷久久| 六月丁香婷婷网| 一本到午夜92版免费福利| 黄乱色伦短篇小说h| 一区二区在线观看高清| 亚洲高清免费视频| 免费在线看黄色| 影音先锋色偷偷米奇四色|