自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Spring Boot打包成的可執(zhí)行jar ,為什么不能被其他項(xiàng)目依賴?

開(kāi)發(fā) 后端
大伙有這樣的疑問(wèn),就是因?yàn)檫€沒(méi)搞清楚可執(zhí)行 jar 和普通 jar 到底有什么區(qū)別?今天松哥就和大家來(lái)聊一聊這個(gè)問(wèn)題。

 前兩天被人問(wèn)到這樣一個(gè)問(wèn)題:

“松哥,為什么我的 Spring Boot 項(xiàng)目打包成的 jar ,被其他項(xiàng)目依賴之后,總是報(bào)找不到類的錯(cuò)誤?”

大伙有這樣的疑問(wèn),就是因?yàn)檫€沒(méi)搞清楚可執(zhí)行 jar 和普通 jar 到底有什么區(qū)別?今天松哥就和大家來(lái)聊一聊這個(gè)問(wèn)題。

多了一個(gè)插件

Spring Boot 中默認(rèn)打包成的 jar 叫做 可執(zhí)行 jar,這種 jar 不同于普通的 jar,普通的 jar 不可以通過(guò) java-jar xxx.jar 命令執(zhí)行,普通的 jar 主要是被其他應(yīng)用依賴, SpringBoot 打成的 jar 可以執(zhí)行,但是不可以被其他的應(yīng)用所依賴,即使強(qiáng)制依賴,也無(wú)法獲取里邊的類。但是可執(zhí)行 jar 并不是 Spring Boot 獨(dú)有的,Java 工程本身就可以打包成可執(zhí)行 jar 。

有的小伙伴可能就有疑問(wèn)了,既然同樣是執(zhí)行 mvnpackage 命令進(jìn)行項(xiàng)目打包,為什么 Spring Boot 項(xiàng)目就打成了可執(zhí)行 jar ,而普通項(xiàng)目則打包成了不可執(zhí)行 jar 呢?

這我們就不得不提 Spring Boot 項(xiàng)目中一個(gè)默認(rèn)的插件配置 spring-boot-maven-plugin ,這個(gè)打包插件存在 5 個(gè)方面的功能,從插件命令就可以看出:

五個(gè)功能分別是:

  •  build-info:生成項(xiàng)目的構(gòu)建信息文件 build-info.properties
  •  repackage:這個(gè)是默認(rèn) goal,在 mvnpackage 執(zhí)行之后,這個(gè)命令再次打包生成可執(zhí)行的 jar,同時(shí)將 mvnpackage 生成的 jar 重命名為 *.origin
  •  run:這個(gè)可以用來(lái)運(yùn)行 Spring Boot 應(yīng)用
  •  start:這個(gè)在 mvn integration-test 階段,進(jìn)行 SpringBoot 應(yīng)用生命周期的管理
  •  stop:這個(gè)在 mvn integration-test 階段,進(jìn)行 SpringBoot 應(yīng)用生命周期的管理

這里功能,默認(rèn)情況下使用就是 repackage 功能,其他功能要使用,則需要開(kāi)發(fā)者顯式配置。

打包

repackage 功能的 作用,就是在打包的時(shí)候,多做一點(diǎn)額外的事情:

  1.  首先 mvnpackage 命令 對(duì)項(xiàng)目進(jìn)行打包,打成一個(gè) jar,這個(gè) jar 就是一個(gè)普通的 jar,可以被其他項(xiàng)目依賴,但是不可以被執(zhí)行
  2.  repackage 命令,對(duì)第一步 打包成的 jar 進(jìn)行再次打包,將之打成一個(gè) 可執(zhí)行 jar ,通過(guò)將第一步打成的 jar重命名為 *.original 文件

舉個(gè)例子:

對(duì)任意一個(gè) Spring Boot 項(xiàng)目進(jìn)行打包,可以執(zhí)行 mvnpackage 命令,也可以直接在 IDEA 中點(diǎn)擊 package,如下 :

打包成功之后, target 中的文件如下:

這里有兩個(gè)文件,第一個(gè) restful-0.0.1-SNAPSHOT.jar 表示打包成的可執(zhí)行 jar ,第二個(gè) restful-0.0.1-SNAPSHOT.jar.original 則是在打包過(guò)程中 ,被重命名的 jar,這是一個(gè)不可執(zhí)行 jar,但是可以被其他項(xiàng)目依賴的 jar。通過(guò)對(duì)這兩個(gè)文件的解壓,我們可以看出這兩者之間的差異。

兩種 jar 的比較

可執(zhí)行 jar 解壓之后,目錄如下:

可以看到,可執(zhí)行 jar 中,我們自己的代碼是存在 于 BOOT-INF/classes/ 目錄下,另外,還有一個(gè) META-INF 的目錄,該目錄下有一個(gè) MANIFEST.MF 文件,打開(kāi)該文件,內(nèi)容如下:   

  1. Manifest-Version: 1.0  
  2.    Implementation-Title: restful  
  3.    Implementation-Version: 0.0.1-SNAPSHOT  
  4.    Start-Class: org.javaboy.restful.RestfulApplication  
  5.    Spring-Boot-Classes: BOOT-INF/classes/  
  6.    Spring-Boot-Lib: BOOT-INF/lib/  
  7.    Build-Jdk-Spec: 1.8  
  8.    Spring-Boot-Version: 2.1.6.RELEASE  
  9.    Created-By: Maven Archiver 3.4.0  
  10.    Main-Class: org.springframework.boot.loader.JarLauncher 

可以看到,這里定義了一個(gè) Start-Class,這就是可執(zhí)行 jar 的入口類, Spring-Boot-Classes 表示我們自己代碼編譯后的位置, Spring-Boot-Lib 則表示項(xiàng)目依賴的 jar 的位置。

換句話說(shuō),如果自己要打一個(gè)可執(zhí)行 jar 包的話,除了添加相關(guān)依賴之外,還需要配置 META-INF/MANIFEST.MF 文件。

這是可執(zhí)行 jar 的結(jié)構(gòu),那么不可執(zhí)行 jar 的結(jié)構(gòu)呢?

我們首先將默認(rèn)的后綴 .original 除去,然后給文件重命名,重命名完成,進(jìn)行解壓:

解壓后可以看到,不可執(zhí)行 jar 根目錄就相當(dāng)于我們的 classpath,解壓之后,直接就能看到我們的代碼,它也有 META-INF/MANIFEST.MF 文件,但是文件中沒(méi)有定義啟動(dòng)類等。   

  1. Manifest-Version: 1.0  
  2.    Implementation-Title: restful  
  3.    Implementation-Version: 0.0.1-SNAPSHOT  
  4.    Build-Jdk-Spec: 1.8  
  5.    Created-By: Maven Archiver 3.4.0 

注意

這個(gè)不可以執(zhí)行 jar 也沒(méi)有將項(xiàng)目的依賴打包進(jìn)來(lái)。

從這里我們就可以看出,兩個(gè) jar ,雖然都是 jar 包,但是內(nèi)部結(jié)構(gòu)是完全不同的,因此一個(gè)可以直接執(zhí)行,另一個(gè)則可以被其他項(xiàng)目依賴。

一次打包兩個(gè) jar

一般來(lái)說(shuō),Spring Boot 直接打包成可執(zhí)行 jar 就可以了,不建議將 Spring Boot 作為普通的 jar 被其他的項(xiàng)目所依賴。如果有這種需求,建議將被依賴的部分,單獨(dú)抽出來(lái)做一個(gè)普通的 Maven 項(xiàng)目,然后在 Spring Boot 中引用這個(gè) Maven 項(xiàng)目。

如果非要將 Spring Boot 打包成一個(gè)普通 jar 被其他項(xiàng)目依賴,技術(shù)上來(lái)說(shuō),也是可以的,給 spring-boot-maven-plugin 插件添加如下配置:   

  1. <build>  
  2.         <plugins>  
  3.             <plugin>  
  4.                 <groupId>org.springframework.boot</groupId>  
  5.                 <artifactId>spring-boot-maven-plugin</artifactId>  
  6.                 <configuration>  
  7.                     <classifier>exec</classifier>  
  8.                 </configuration>  
  9.             </plugin>  
  10.         </plugins>  
  11.     </build> 

配置的 classifier 表示可執(zhí)行 jar 的名字,配置了這個(gè)之后,在插件執(zhí)行 repackage 命令時(shí),就不會(huì)給 mvnpackage 所打成的 jar 重命名了,所以,打包后的 jar 如下:

第一個(gè) jar 表示可以被其他項(xiàng)目依賴的 jar ,第二個(gè) jar 則表示一個(gè)可執(zhí)行 jar。

好了,關(guān)于 Spring Boot 中 jar 的問(wèn)題,我們就說(shuō)這么多。 

 

責(zé)任編輯:龐桂玉 來(lái)源: Java編程
相關(guān)推薦

2021-01-14 22:17:09

PythonLinux工具

2012-12-18 11:55:38

2009-07-08 15:18:00

JAR文件包jar命令

2022-04-24 11:06:54

SpringBootjar代碼

2017-02-07 20:49:15

2023-03-08 09:48:08

SpringWAR文件JAR文件

2011-11-10 10:39:21

蘋(píng)果Siri

2023-05-29 08:43:46

SpringJava

2022-05-20 08:55:02

py文件exepython

2022-01-11 09:59:23

Python關(guān)機(jī)程序文件

2020-05-07 16:30:32

Spring BootJava

2011-09-13 17:30:41

Eclipse And

2024-02-20 08:51:56

PythonFlask工具

2009-06-10 20:44:18

netbeans 打包JAR文件

2023-12-18 09:21:22

開(kāi)發(fā)靜態(tài)編譯Linux

2020-08-14 10:40:35

RestTemplatRetrofitJava

2021-05-07 07:03:33

Spring打包工具

2024-03-08 09:34:35

JpackageJarJava

2020-06-30 14:25:28

MavenGradle Spring

2018-01-10 13:58:20

點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)