關(guān)于SpringBoot 打包成的可執(zhí)行jar不能被其他項目依賴的問題
Spring Boot 項目打包成的 jar ,被其他項目依賴之后,總是報找不到類的錯誤?
大伙有這樣的疑問,就是因為還沒搞清楚可執(zhí)行 jar 和普通 jar 到底有什么區(qū)別?
一、Springboot 項目的默認(rèn)插件配置
Spring Boot 中默認(rèn)打包成的 jar 叫做 可執(zhí)行 jar,這種 jar 不同于普通的 jar,普通的 jar 不可以通過 java -jar xxx.jar
命令執(zhí)行,
普通的 jar 主要是被其他應(yīng)用依賴,Spring Boot 打成的 jar 可以執(zhí)行,但是不可以被其他的應(yīng)用所依賴,即使強(qiáng)制依賴,也無法獲取里邊的類。但是可執(zhí)行 jar 并不是 Spring Boot 獨有的,Java 工程本身就可以打包成可執(zhí)行 jar 。
有的小伙伴可能就有疑問了,既然同樣是執(zhí)行 mvn package
命令進(jìn)行項目打包,為什么 Spring Boot 項目就打成了可執(zhí)行 jar ,而普通項目則打包成了不可執(zhí)行 jar 呢?
這我們就不得不提 Spring Boot 項目中一個默認(rèn)的插件配置 spring-boot-maven-plugin
,這個打包插件存在 5 個方面的功能,從插件命令就可以看出:
五個功能分別是:
(1)
build-info
:生成項目的構(gòu)建信息文件build-info.properties
(2)repackage
:這個是默認(rèn) goal,在mvn package
執(zhí)行之后,這個命令再次打包生成可執(zhí)行的 jar,同時將mvn package
生成的 jar 重命名為*.origin
(3)run:這個可以用來運(yùn)行 Spring Boot 應(yīng)用
(4)start:這個在mvn integration-test
階段,進(jìn)行 Spring Boot 應(yīng)用生命周期的管理
(5)stop:這個在mvn integration-test
階段,進(jìn)行 Spring Boot 應(yīng)用生命周期的管理
這里功能,默認(rèn)情況下使用就是 repackage 功能,其他功能要使用,則需要開發(fā)者顯式配置。
二、打包
repackage 功能的 作用,就是在打包的時候,多做一點額外的事情:
(1)首先
mvn package
命令 對項目進(jìn)行打包,打成一個 jar,這個 jar 就是一個普通的 jar,可以被其他項目依賴,但是不可以被執(zhí)行
(2)repackage
命令,對第一步 打包成的 jar 進(jìn)行再次打包,將之打成一個 可執(zhí)行 jar ,通過將第一步打成的 jar 重命名為*.original
文件
舉個例子:
對任意一個 Spring Boot 項目進(jìn)行打包,可以執(zhí)行 mvn package
命令,也可以直接在 IDEA 中點擊 package
,如下 :
這里有兩個文件,第一個 restful-0.0.1-SNAPSHOT.jar
表示打包成的可執(zhí)行 jar ,第二個 restful-0.0.1-SNAPSHOT.jar.original
則是在打包過程中 ,被重命名的 jar,這是一個不可執(zhí)行 jar,但是可以被其他項目依賴的 jar。通過對這兩個文件的解壓,我們可以看出這兩者之間的差異。
三、兩種 jar 包的比較
1. admin-0.0.1-SNAPSHOT.jar 可執(zhí)行的 jar 結(jié)構(gòu)
可以看到,可執(zhí)行 jar 中,我們自己的代碼是存在 于 BOOT-INF/classes/
目錄下,另外,還有一個 META-INF
的目錄,該目錄下有一個 MANIFEST.MF
文件,打開該文件,內(nèi)容如下:
Manifest-Version: 1.0 Created-By: Maven Jar Plugin 3.2.0 Build-Jdk-Spec: 14 Implementation-Title: admin Implementation-Version: 0.0.1-SNAPSHOT Main-Class: org.springframework.boot.loader.JarLauncher Start-Class: org.yolo.admin.AdminApplication Spring-Boot-Version: 2.3.4.RELEASE Spring-Boot-Classes: BOOT-INF/classes/ Spring-Boot-Lib: BOOT-INF/lib/ Spring-Boot-Classpath-Index: BOOT-INF/classpath.idx
可以看到,這里定義了一個 Start-Class
,這就是可執(zhí)行 jar 的入口類,Spring-Boot-Classes
表示我們自己代碼編譯后的位置,Spring-Boot-Lib
則表示項目依賴的 jar 的位置。
換句話說,如果自己要打一個可執(zhí)行 jar 包的話,除了添加相關(guān)依賴之外,還需要配置 META-INF/MANIFEST.MF
文件。
這是可執(zhí)行 jar 的結(jié)構(gòu),那么不可執(zhí)行 jar 的結(jié)構(gòu)呢?
2. admin-0.0.1-SNAPSHOT.jar.original 不可執(zhí)行的 jar 結(jié)構(gòu)
我們首先將默認(rèn)的后綴 .original
除去,然后給文件重命名,重命名完成,進(jìn)行解壓:
解壓后可以看到,不可執(zhí)行 jar 根目錄就相當(dāng)于我們的 classpath
,解壓之后,直接就能看到我們的代碼,它也有 META-INF/MANIFEST.MF
文件,但是文件中沒有定義啟動類等。
Manifest-Version: 1.0 Created-By: Maven Jar Plugin 3.2.0 Build-Jdk-Spec: 14 Implementation-Title: admin Implementation-Version: 0.0.1-SNAPSHOT
注意
這個不可以執(zhí)行 jar 也沒有將項目的依賴打包進(jìn)來。
從這里我們就可以看出,兩個 jar ,雖然都是 jar 包,但是內(nèi)部結(jié)構(gòu)是完全不同的,因此一個可以直接執(zhí)行,另一個則可以被其他項目依賴。
四、一次性打包兩個 jar
一般來說,Spring Boot 直接打包成可執(zhí)行 jar 就可以了,不建議將 Spring Boot 作為普通的 jar 被其他的項目所依賴。如果有這種需求,建議將被依賴的部分,單獨抽出來做一個普通的 Maven 項目,然后在 Spring Boot 中引用這個 Maven 項目
如果非要將 Spring Boot 打包成一個普通 jar 被其他項目依賴,技術(shù)上來說,也是可以的,給 spring-boot-maven-plugin
插件添加如下配置:
<build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <classifier>exec</classifier> </configuration> </plugin> </plugins> </build>
配置的 classifier
表示可執(zhí)行 jar 的名字,配置了這個之后,在插件執(zhí)行 repackage
命令時,就不會給 mvn package
所打成的 jar 重命名了,所以,打包后的 jar 如下:
第一個 jar 表示可以被其他項目依賴的 jar ,第二個 jar 則表示一個可執(zhí)行 jar。
到此這篇關(guān)于關(guān)于SpringBoot 打包成的可執(zhí)行 jar不能被其他項目依賴的問題的文章就介紹到這了,更多相關(guān)SpringBoot 打包成的可執(zhí)行 jar內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決java調(diào)用dll報Unable to load library錯誤的問題
這篇文章主要介紹了解決java調(diào)用dll報Unable to load library錯誤的問題。具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-11-11Myeclipse鏈接Oracle等數(shù)據(jù)庫時lo exception: The Network Adapter coul
今天小編就為大家分享一篇關(guān)于Myeclipse鏈接Oracle等數(shù)據(jù)庫時lo exception: The Network Adapter could not establish the connection,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03Spring Cloud Gateway + Nacos 實現(xiàn)動態(tài)路由
這篇文章主要介紹了Spring Cloud Gateway + Nacos 實現(xiàn)動態(tài)路由的方法,幫助大家實現(xiàn)路由信息的自動更新,感興趣的朋友可以了解下2020-10-10SpringBoot項目中@Test不出現(xiàn)可點擊運(yùn)行的按鈕問題
這篇文章主要介紹了SpringBoot項目中@Test不出現(xiàn)可點擊運(yùn)行的按鈕問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-01-01java提取字符串中數(shù)字string以及獲取字符串中的整數(shù)或小數(shù)
這篇文章主要給大家介紹了關(guān)于java提取字符串中數(shù)字string以及獲取字符串中的整數(shù)或小數(shù)的相關(guān)資料,需要的朋友可以參考下2023-08-08java使用任務(wù)架構(gòu)執(zhí)行任務(wù)調(diào)度示例
在Java 5.0之前啟動一個任務(wù)是通過調(diào)用Thread類的start()方法來實現(xiàn)的,5.0里提供了一個新的任務(wù)執(zhí)行架構(gòu)使你可以輕松地調(diào)度和控制任務(wù)的執(zhí)行,并且可以建立一個類似數(shù)據(jù)庫連接池的線程池來執(zhí)行任務(wù),下面看一個示例2014-01-01