精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

淺析 SpringBoot FatJar 機制的設計與實現

開發 前端
打開 Java 的 jar 文件我們經常可以看到文件中包含著一個META-INF?目錄,這個目錄下會有一些文件,其中必有一個MANIFEST.MF,這個文件描述了該 Jar 文件的很多信息 其中 Main-Class 定義 Jar 文件的入口類,該類必須是一個可執行的類,一旦定義了該屬性即可通過 java -jar xxx.jar來運行該 jar 文件。

一、概述

SpringBoot FatJar 的設計,打破了標準 jar 的結構,在 jar 包內攜帶了其所依賴的 jar 包,通過在標準 jar 包中指定的 Main-Class 的 main 方法啟動后,創建自己的類加載器,來識別、加載、運行其非規范的目錄下的代碼(BOOT-INF/classes/...)和依賴(BOOT-INF/lib/...)。BOOT-INF/classes/ 目錄下有 SpringBoot 上下文的啟動類的 class 文件,自定義類加載器加載這個啟動類后,開始進入 SpringBoot 的上下文中運行我們所寫的程序代碼。執行的流程可概括為:

  1. 通過 java -jar xxx.jar 啟動應用
  2. 執行 xxx.jar 中 META-INF/MANIFEST.MF 里 Main-Class 所指定的 JarLauncher  類的 main 方法
  3.  main 方法中創建自定義的 ClassLoader 即 LaunchedURLClassLoader,并將其設置為線程上下文類加載器
  4. 由 LaunchedURLClassLoader 加載 META-INF/MANIFEST.MF 里 Start-Class 所指定的 SpringBoot 應用的啟動類(在 BOOT-INF/classes/目錄下),調用其 main 方法,開始執行 SpringApplication.run(...)

二、標準的 jar 包結構

打開 Java 的 jar 文件我們經??梢钥吹轿募邪粋€META-INF目錄,這個目錄下會有一些文件,其中必有一個MANIFEST.MF,這個文件描述了該 Jar 文件的很多信息 其中 Main-Class 定義 Jar 文件的入口類,該類必須是一個可執行的類,一旦定義了該屬性即可通過  java -jar xxx.jar來運行該 jar 文件。

在生產環境中,是使用java -jar xxx.jar 的方式來運行 SpringBoot 程序。這種情況下,SpringBoot 應用真實的啟動類并不是我們所定義的帶有 main 方法的啟動類,而是其內置的 JarLauncher 類。查看 SpringBoot 所打成的 fat jar,其 Main-Class 是org.springframework.boot.loader.JarLauncher,這便是微妙之處。

Spring-Boot-Version: 2.1.3.RELEASE
Main-Class: org.springframework.boot.loader.JarLauncher
Start-Class: com.rock.springbootlearn.SpringbootLearnApplication
Spring-Boot-Classes: BOOT-INF/classes/
Spring-Boot-Lib: BOOT-INF/lib/
Build-Jdk: 1.8.0_131

JAR 包中的 MANIFEST.MF 文件詳解以及編寫規范[1]

三、探索 JarLauncher

org.springframework.boot.loader.JarLauncher這個類是哪里來的呢?答案在 spring-boot-loader-***.jar 包中,可找到這個 JarLauncher 類的源碼。在項目中加入 maven 依賴,以便查看源碼和遠程調試。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-loader</artifactId>
</dependency>

圖片圖片

認真比較可以看出,這個 spring-boot-loader 包中的內容與 SpringBoot 的 FatJar 包中的一部分內容幾乎一樣。JarLauncher 在 jar 中的位置如下:

圖片圖片

3.1 只能拷貝出來一份兒

重點重點重點:因 jar 規范要求 Main-Class 所指定的類必須位于 jar 包的頂層目錄下,即 org.springframework.boot.loader.JarLauncher 這個 org 必須位于 jar 包中的第一級目錄,不能放置在其他的目錄下。所以 所以 所以(重點)只能將 spring-boot-loader 這個 jar 包的內容拷貝出來,而不是整個 jar 直接放置于執行 Jar 中。

3.2 攜帶程序所依賴的 jar 而非僅 class

圖片圖片

上邊 JarLauncher 的這個 org.springframework.xx 以及 META-INF 這兩個目錄是符合 jar 包規范的。但是 BOOT-INF 這個目錄里邊有點像我們開發中的一些用法:

  • 依賴的  jar 包在 lib 目錄下
  • 但按照 jar 包規范 jar 中不能有 jar 包的情況下
  • 程序.class 文件在 classes 目錄下
  • 但 xxx.class 文件應該按照 org.springframework.xx 這樣放置在 jar 中的根目錄中

所以classes 和 lib 你也能意識到,這個設計是獨特的。早期 jar 包內攜帶依賴是采用如 maven-shade-plugin 的做法,把依賴的 class 文件拷貝到目標 jar 中,但也會造成重名(全限定名)的類會出現覆蓋的情況。后來 SpringBoot 為了避免提取覆蓋的情況,修改了打包機制,放棄了maven-shade-plugin那種拷貝 class 的方式,調整為依賴原始 jar 包;這同時意味著改變了 jar 標準的運行機制,那么要想讓classes和lib中代碼能夠正常運行,你試想一下如果沒有自定義的類加載器來加載這些類文件,可以嘛?答案是:不可以,需要自定義類加載器。

四、 自定義類加載器的運行機制

通常自定義類加載器完成資源加載的核心邏輯為:

  1. 指定資源
  2. 指定委托關系
  3. 指定線程上下文類加載器
  4. 調用邏輯入口方法

所以 SpringBoot FatJar 中自定義 ClassLoader 中的核心邏輯也是如此。

4.1 指定資源

構造方法中基于 jar 包的文件系統信息,構造 Archive 對象

public ExecutableArchiveLauncher() {
 this.archive = createArchive();
}

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

采集 jar 包中的 classes和lib目錄下的歸檔文件。后邊創建 ClassLoader 的時候作為參數傳入

@Override
protected List<Archive> getClassPathArchives() throws Exception {
 List<Archive> archives = new ArrayList<>(
   this.archive.getNestedArchives(this::isNestedArchive));
 postProcessClassPathArchives(archives);
 return archives;
}

protected boolean isNestedArchive(Archive.Entry entry) {
 if (entry.isDirectory()) {
  return entry.getName().equals(BOOT_INF_CLASSES);
 }
 return entry.getName().startsWith(BOOT_INF_LIB);
}

4.2 創建自定義 ClassLoader

protected void launch(String[] args) throws Exception {
 JarFile.registerUrlProtocolHandler();
        //創建類加載器, 并指定歸檔文件
 ClassLoader classLoader = createClassLoader(getClassPathArchives());
 launch(args, getMainClass(), classLoader);
}
//創建類加載器, 將歸檔文件轉換為URL
protected ClassLoader createClassLoader(List<Archive> archives) throws Exception {
 List<URL> urls = new ArrayList<>(archives.size());
 for (Archive archive : archives) {
  urls.add(archive.getUrl());
 }
 return createClassLoader(urls.toArray(new URL[0]));
}
//父加載器是AppClassLoader
protected ClassLoader createClassLoader(URL[] urls) throws Exception {
        //getClass().getClassLoader() 是系統類加載器,因為默認情況下main方法所在類是由SystemClassLoader加載的,默認情況下是AppClassLoader.
 return new LaunchedURLClassLoader(urls, getClass().getClassLoader());
}

4.3 設置線程上下文類加載器,調用程序中的 main 方法

public static void main(String[] args) throws Exception {
 new JarLauncher().launch(args);
}
protected void launch(String[] args, String mainClass, ClassLoader classLoader)
  throws Exception {
        //設置線程上下文類加載器
 Thread.currentThread().setContextClassLoader(classLoader);
 //調用MANIFEST.MF 中配置的Start-Class: xxx的main方法,還帶入了參數
        createMainMethodRunner(mainClass, args, classLoader).run();

本文轉載自微信公眾號「架構染色」,可以通過以下二維碼關注。轉載本文請聯系【架構染色】公眾號作者。


責任編輯:武曉燕 來源: 架構染色
相關推薦

2020-07-19 10:26:47

Kubernetes數據結構

2012-09-25 13:24:29

ERP系統

2016-09-29 22:54:55

2009-08-10 14:38:29

ASP.NET組件設計

2009-06-11 17:03:29

Java線程

2016-01-08 10:06:52

2017-04-12 11:46:46

前端瀏覽器渲染機制

2011-05-19 10:57:45

DNSSEC密鑰加密

2023-10-27 08:20:12

springboot微服務

2021-08-30 09:44:47

Kubelet機制驅逐

2009-08-05 16:59:55

ASP.NET組件設計

2011-04-21 10:01:37

WTL CString

2022-06-01 16:01:58

MySQL內存管理系統

2009-06-23 14:15:00

Java垃圾回收

2022-08-28 20:50:29

算法模型機器學習

2018-03-14 08:39:40

2009-08-04 14:18:49

ASP.NET郵件列表

2010-06-24 15:30:47

HART協議

2020-12-31 05:57:36

DockerSpringBootMaven

2009-06-29 17:39:31

JSP設計模式
點贊
收藏

51CTO技術棧公眾號

国产91在线看| 欧美区一区二| 欧美无砖专区一中文字| 国产欧美综合一区| 亚洲精品一区二区三区区别| 国产视频一区三区| 久久精品久久久久久| 在线观看一区二区三区四区| 欧美自拍电影| 亚洲自拍偷拍网站| 日韩av图片| 可以免费看毛片的网站| 久久久久久穴| 欧美高清视频在线| 国产又黄又粗又猛又爽的| 9l亚洲国产成人精品一区二三| 色综合天天狠狠| 黄色三级中文字幕| 波多野结衣在线影院| 成人天堂资源www在线| 国产九九精品视频| 亚洲第一在线播放| 亚洲一级淫片| 自拍偷拍亚洲欧美| 亚洲最大免费视频| 日韩一级淫片| 欧美精品三级在线观看| 国产精品69页| 欧美男男tv网站在线播放| 亚洲欧美日韩系列| 亚洲午夜高清视频| 飘雪影院手机免费高清版在线观看| 国产剧情一区二区三区| 国产精品视频xxx| 欧美一区二区三区网站| 最新亚洲激情| 欧美高清在线视频观看不卡| 无码人中文字幕| 红桃成人av在线播放| 亚洲精品国产精品国自产在线 | 国产在线拍揄自揄拍无码| 日韩精品123| gogo大胆日本视频一区| 国产精品免费在线播放| 亚洲精品一区二区三区不卡| 国产一区二区三区四区五区美女| 国产日本欧美在线观看| 中文字幕乱码视频| 全国精品久久少妇| 国产精品视频在线播放| 免费精品一区二区| 日本不卡一区二区| 国产精品久久久久久久app| 四虎影院在线免费播放| 香蕉视频成人在线观看| 国产aⅴ夜夜欢一区二区三区| 午夜婷婷在线观看| 玖玖玖国产精品| 国产精品高清免费在线观看| 中文无码精品一区二区三区| 久久97超碰色| 91中文字幕在线观看| av一区二区三| 成人午夜电影网站| 久久一区免费| 大胆av不用播放器在线播放| 国产精品每日更新| 国产日韩欧美大片| 美洲精品一卡2卡三卡4卡四卡| 亚洲蜜臀av乱码久久精品| 久久久久久av无码免费网站下载| 色呦呦网站在线观看| 亚洲成a人片综合在线| 国产一级爱c视频| 免费看av不卡| 欧美久久久久久久久| 激情久久综合网| 综合久久成人| 亚洲精选中文字幕| 日本视频在线免费| 欧美成熟视频| 欧美亚洲视频在线看网址| 无码人妻黑人中文字幕| 久久99国产精品尤物| 国产成人女人毛片视频在线| 日韩欧美亚洲系列| 国产精品高清亚洲| 国产情侣第一页| 偷拍视频一区二区三区| 欧美电影在线免费观看| 91丨porny丨对白| heyzo久久| 欧美激情小视频| 亚洲大尺度在线观看| 国产成人av一区| 日本精品一区二区三区高清 久久| 一区二区高清不卡| 亚洲一区二区av电影| 午夜视频在线瓜伦| 欧美在线在线| 国产一区二区三区18 | 五月激情综合网| 激情五月婷婷久久| 日韩影片在线观看| 最近2019年日本中文免费字幕| 久久婷婷国产麻豆91| 日韩avvvv在线播放| www.成人av| 95在线视频| 粉嫩老牛aⅴ一区二区三区| 一级黄色录像在线观看| 噜噜噜天天躁狠狠躁夜夜精品 | 久久丁香综合五月国产三级网站| 粉嫩高清一区二区三区精品视频| 第一福利在线| 狠狠久久亚洲欧美专区| 性xxxxxxxxx| 日韩av自拍| 日本国产高清不卡| 欧美一级淫片aaaaaa| 国产精品美女久久久久aⅴ国产馆| 可以在线看的av网站| 精品国产第一国产综合精品| 国产一区二区日韩| 秋霞精品一区二区三区| 成人av综合一区| 日本a级片在线观看| 激情久久99| 亚洲午夜未删减在线观看 | 中国女人特级毛片| 99在线观看免费视频精品观看| 亚洲在线视频福利| 黄色国产网站在线播放| 精品视频一区三区九区| 无码人妻精品一区二区三区温州| 欧美日韩精品一本二本三本 | 99riav国产| 国产精品美女久久久久aⅴ | 神马久久久久久| 波多野结衣二区三区| 久久综合九色欧美综合狠狠| 野外做受又硬又粗又大视频√| gogo大尺度成人免费视频| 少妇高潮 亚洲精品| 中文在线字幕免费观| 国产亚洲一区字幕| 国产福利一区视频| 国产成人一区二区三区影院| 国产99久久精品一区二区 夜夜躁日日躁 | 日韩精品一区二区三区swag| 午夜国产福利一区二区| 国内精品伊人久久久久av一坑| 亚洲精品一区二区三区樱花 | 成人在线视频电影| 日韩伦理av| 亚洲电影av在线| 日本最新中文字幕| 91麻豆蜜桃一区二区三区| 北条麻妃在线观看| 国产精品一区二区三区av麻| 国产精品久久久久久久久久久久久久 | 999免费视频| 一区二区三区视频在线看| 丰满饥渴老女人hd| 一本色道久久综合亚洲精品高清 | 91福利视频在线| 国产性猛交xx乱| 久久精品国产第一区二区三区| 一区二区精品视频| 美国十次综合久久| 午夜精品视频在线| 国产在线资源| 91麻豆精品国产91久久久资源速度| 四虎精品免费视频| av在线不卡观看免费观看| av免费网站观看| 99久久亚洲精品蜜臀| αv一区二区三区| 日本综合字幕| 久久久av电影| 天天操天天干天天| 欧美色倩网站大全免费| 久久久久久久久久一区二区三区| 久久综合九色综合97婷婷| 亚洲精品久久久中文字幕| 欧美精品99| 美日韩免费视频| 精品国产伦一区二区三区观看说明| 久久人人爽人人| 超碰免费在线| 亚洲精品国产综合久久| 成人黄色三级视频| 亚洲aaa精品| 五月综合色婷婷| 26uuu另类欧美| 国产探花一区二区三区| 日韩精品亚洲一区二区三区免费| 美女在线免费视频| 欧美久久精品一级c片| 国产精品欧美久久| 亚洲国产91视频| 日韩av第一页| mm视频在线视频| 久久久999国产| 丁香婷婷在线| 亚洲国内精品视频| 国产av无码专区亚洲av| 91国产免费观看| 日本中文字幕在线免费观看| 亚洲三级电影网站| 久久久久无码精品国产sm果冻 | 国产亚洲综合精品| 男人的天堂视频在线| 精品freesex老太交| 国产原创精品| 亚洲精品一区二区三区中文字幕| 国产精品免费视频久久久| 小视频免费在线观看| 久久久久久91| 三级网站视频在在线播放| 在线视频精品一| 欧美高清成人| 亚洲精品国产精品国自产观看浪潮| 精品国产黄色片| 欧美区一区二区三区| 国产成人无码专区| 色综合久久88色综合天天| 日韩 国产 在线| 亚洲亚洲精品在线观看| 日日骚一区二区三区| 国产精品毛片无遮挡高清| 成年人在线免费看片| 国产拍欧美日韩视频二区| 久久久久久久久免费看无码| 成人aaaa免费全部观看| 午夜福利三级理论电影| 国产成人av电影在线播放| 人妻精油按摩bd高清中文字幕| 激情图片小说一区| 51自拍视频在线观看| 韩国三级中文字幕hd久久精品| 杨幂毛片午夜性生毛片| 免费在线成人网| 性chinese极品按摩| 日本一区中文字幕| 欧美午夜aaaaaa免费视频| 日本成人在线不卡视频| 国产区二区三区| 久久国产人妖系列| 99中文字幕在线| 国产精品亚洲综合一区在线观看| 波多野结衣免费观看| 粉嫩一区二区三区性色av| 精品人妻一区二区三| 国产精品456| 国产香蕉精品视频| 95精品视频在线| 欧美成人国产精品一区二区| 欧美国产精品一区| 情侣偷拍对白清晰饥渴难耐| 亚洲欧美日韩中文播放 | 阿v免费在线观看| 中文字幕在线观看日韩| 欧美13一16娇小xxxx| 久久精品一本久久99精品| fc2ppv国产精品久久| 欧美多人爱爱视频网站| 涩涩网在线视频| 国产精品福利无圣光在线一区| 欧美黄页在线免费观看| 亚洲xxxx做受欧美| 欧美a级网站| 日韩视频在线播放| 91精品亚洲| 青青草国产精品视频| 视频一区中文字幕国产| 亚洲一区精品视频在线观看| 成人午夜电影久久影院| 亚洲午夜福利在线观看| 中文字幕在线视频一区| 国产一级黄色av| 日韩欧美中文字幕在线播放| 一区二区三区免费观看视频| 欧美精品一区二区久久久| 成年人免费在线视频| 色综合视频一区中文字幕| 午夜无码国产理论在线| 亚洲自拍在线观看| 色综合中文网| 黄网站色视频免费观看| 日日夜夜免费精品视频| 亚洲精品久久一区二区三区777| 久久久www免费人成精品| 校园春色 亚洲| 在线观看日产精品| 免费观看的毛片| 日韩中文视频免费在线观看| 国产网站在线| 亚洲在线第一页| av在线不卡顿| 国产肥臀一区二区福利视频| 国精产品一区一区三区mba视频| 久久久久麻豆v国产精华液好用吗| 国产精品美女久久久久aⅴ| 天天爽夜夜爽夜夜爽精品| 欧美精品aⅴ在线视频| 青青草视频在线免费观看| 欧美成人sm免费视频| h1515四虎成人| 久久亚洲精品欧美| 国内精品嫩模av私拍在线观看| 四季av一区二区三区| 久久精品欧美日韩| 国产91精品一区| 亚洲а∨天堂久久精品9966| 国产原创精品视频| 国产精品久久不能| 性人久久久久| www.av蜜桃| 国产精品18久久久久久久久久久久| 性欧美一区二区| 日韩欧美亚洲成人| 少妇av在线播放| 色综合91久久精品中文字幕 | 国产精品推荐精品| 午夜视频精品| 国产精品久久久久久久av福利| 久久免费精品国产久精品久久久久 | 亚洲男女视频在线观看| 久久国产精品偷| 91成人在线网站| 亚洲一区bb| 久久se这里有精品| 波兰性xxxxx极品hd| 欧美日韩在线观看一区二区 | 一区二区三区欧美在线观看| 亚洲一区二区三区高清视频| 一本色道久久88综合亚洲精品ⅰ | 人妻精品久久久久中文| 懂色av中文一区二区三区天美| 男人天堂av网| 久久男人av资源网站| 国产精品2023| 少妇人妻无码专区视频| 成人av网址在线| 亚洲黄色三级视频| 亚洲精品一区二区网址| 日本三级一区| 欧美色图亚洲自拍| 久久天堂成人| 少妇视频在线播放| 欧美精选在线播放| 99福利在线| 国产乱码一区| 久久久777| 久久日免费视频| 欧美精品日韩精品| 丝袜综合欧美| 精品蜜桃一区二区三区| 久久激情中文| 国产成人免费在线观看视频| 在线电影欧美成精品| 性欧美1819sex性高清大胸| 国产一区二区三区四区五区加勒比| 亚洲精品一二| 老熟妇一区二区| 制服丝袜成人动漫| 97久久人人超碰caoprom| 久久久久se| 久久精品99国产精品日本| 欧美日韩三级在线观看| 亚洲精品黄网在线观看| 欧美色网一区| 特级西西444| 91影院在线观看| 中文字幕网址在线| 九九视频直播综合网| 外国成人在线视频| 国内国产精品天干天干| 性做久久久久久免费观看| 国产三级在线免费| 91手机在线观看| 久久婷婷av| 久久这里只有精品免费| 亚洲午夜久久久久久久| 亚洲图色一区二区三区| 欧美精品第三页| 亚洲精品老司机| 精品无吗乱吗av国产爱色| 91久久精品日日躁夜夜躁国产| 99热这里只有精品8| 亚洲熟女少妇一区二区| 亚洲第一区第一页| 日韩成人在线一区| 91专区在线观看| 18成人在线观看| 欧美日韩国产亚洲沙发| 亚洲va电影大全| 日韩av电影一区| 800av免费在线观看|