使用springboot 打包插件去除jar包瘦身
1、pom文件配置
1.1 添加maven-dependency-plugin插件用於將引用的jar包拷貝到指定的路徑
便於後續tomcat啟動指定依賴包路徑
<!--拷貝依賴到jar外面的lib目錄--> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-dependency-plugin</artifactId> <executions> <execution> <id>copy</id> <phase>package</phase> <goals> <goal>copy-dependencies</goal> </goals> <configuration> <!--指定的依賴路徑--> <outputDirectory> ${project.build.directory}/lib </outputDirectory> </configuration> </execution> </executions> </plugin>
使用這個插件構建完之後的目錄結構多瞭一個lib目錄(即上述配置的outputDirectory指定的路徑),裡面是依賴的jar包:
1.2 springboot項目使用spring-boot-maven-plugin打包插件
<plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <configuration> <executable>true</executable> <layout>ZIP</layout> <mainClass> com.iasp.BasicStarter </mainClass> <!--隻包含自己--> <includes> <include> <groupId>${groupId}</groupId> <artifactId>${artifactId}</artifactId> </include> <!--或者--> <!--依賴jar不打進項目jar包中--> <!--<include> <groupId>nothing</groupId> <artifactId>nothing</artifactId> </include>--> </includes> <!--不包含哪些--> <!--<excludeGroupIds>--> <!--com.hundsun.jrescloud,--> <!--org.springframework.boot,--> <!--org.springframework--> <!--</excludeGroupIds>--> </configuration> <executions> <execution> <goals> <goal>repackage</goal> </goals> </execution> </executions> </plugin>
配置上述打包之後就會排除相應的jar包,使得由插件打成的Flat jar包大小變小,便於上傳服務器發佈,效果如下,BOOT-INF目錄下的lib目錄沒有瞭:
原先打成的jar包裡的結構為
然後在啟動項目時指定jar包路徑-Dloader.path=”../lib”,這樣就可以達到瘦身效果瞭,其中依賴放在D:develop/shared/fjar目錄下,執行運行命令
java -Dloader.path="D:develop/shared/fjar" -jar mytest.jar
附註:另外一種啟動方案是可以不加-Dloader.path=”D:develop/shared/fjar”來指定路徑,直接使用如下指令啟動
java -jar mytest.jar
使用上述啟動的話需要添加maven-jar-plugin插件,配置<classpathPrefix>屬性,另外在處理一些讀取可執行jar中的文件時,可以使用maven-jar-plugin插件替換spring-boot-maven-plugin進行打包操作
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-jar-plugin</artifactId> <configuration> <archive> <manifest> <!--addClasspath表示需要加入到類構建路徑--> <addClasspath>true</addClasspath> <!--classpathPrefix指定生成的Manifest文件中Class-Path依賴lib前面都加上路徑,構建出lib/xx.jar--> <classpathPrefix>lib/</classpathPrefix> <mainClass>com.common.util.CommonUtilsApplication</mainClass> </manifest> </archive> </configuration> </plugin>
上述插件效果就是在打成的包裡META_INF目錄下的MANIFEST.MF文件裡增加Class-path對應jar,這樣在後面應用剛啟動時就會根據Class-Path的隻去加載需要的版本依賴(解決在共享目錄裡存在多版本加載引用沖突問題),這個效果就等效加參數-classpath xxx(具體的jar)。
此時就是將需要的jar目錄lib放在和要運行的xxx.jar同級目錄即可,啟動時就可以不加-Dloader.path參數瞭,如果lib目錄和要運行的xxx.jar不在同級目錄的話,則需要使用-Dloader.path來啟動
如下:在同一級目錄啟動
不在同一級目錄啟動:
其中-Dloader.path可以指定多個目錄,這樣在存在多個微服務情況下可將一些公共用到的jar放在一個共享目錄中,每個微服務獨有的jar可以放在微服務私有的目錄下(解決jar版本沖突問題),示例如下:
註意:
1、使用-Dloader.path需要在打包的時候增加<layout>ZIP</layout>,不指定的話-Dloader.path不生效
對於多個微服務瘦身打包建議使用maven-jar-plugin打包,避免因為spring-boot-maven-plugin打包機制導致的一些應用啟動問題(已踩坑)
2、若存在不同版本依賴:
比如項目A依賴Y庫的1.0版本,項目B依賴Y庫的2.0版本,那麼可能會出現版本依賴沖突(兩個版本不兼容的情況下),解決方案:
2.1、能做到版本一致就保持使用同一個版本,保證版本一致。可以使用maven的版本依賴管理進行處理,即在父pom文件使用<dependencyManagement>統一管理依賴版本
2.2、讓項目各自依賴所需的版本並打進war包中,把其他同版本的jar包放在同一個共享包下
測試發現依賴在查找時從上往下找,匹配到就用第一個,如下圖會使用comm-0.0.1.jar版本的
附註:
使用spring-boot-maven-plugin插件,會將依賴的jar包全部打包進去,這樣就可以直接運行生成的 JAR 包,簡化瞭我們開發操作。
使用spring-boot-maven-plugin插件如果不指定程序主運行入口類的話默認為Main-Class: org.springframework.boot.loader.JarLauncher
這個可以自定義執行主入口類,有以下幾種方式:
1.POM繼承spring-boot-starter-parent
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.1.9.RELEASE</version> <relativePath/> <!-- lookup parent from repository --> </parent> <properties> <!-- The main class to start by executing java -jar --> <start-class>ccom.notes.JavaNotesApplication</start-class> </properties>
2.POM不是繼承spring-boot-starter-parent時需指定
<plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <version>2.1.9.RELEASE</version> <configuration> <mainClass>com.notes.JavaNotesApplication</mainClass> <layout>ZIP</layout> </configuration> <executions> <execution> <goals> <goal>repackage</goal> </goals> </execution> </executions> </plugin>
3.POM不是繼承spring-boot-starter-paren,且使用maven-jar-plugin插件來指定執行的類
<plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-jar-plugin</artifactId> <configuration> <archive> <manifest> <addClasspath>true</addClasspath> <useUniqueVersions>false</useUniqueVersions> <classpathPrefix>lib/</classpathPrefix> <mainClass>com.notes.JavaNotesApplication</mainClass> </manifest> <manifestEntries> <version>${project.version}</version> </manifestEntries> </archive> </configuration> </plugin>
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- springboot maven 打包插件介紹及註意事項說明
- spring boot項目沒有mainClass如何實現打包運行
- SpringBoot分離打Jar包的兩種配置方式
- Maven如何打入依賴中指定的部分jar包
- Maven打包時如何指定啟動類