Spring Boot項目可以通過spring-boot-maven-plugin插件打包生成一個可執(zhí)行的jar包先誉,這樣可以脫離web容器(例如tomcat)直接運行。但默認(rèn)情況下spring-boot-maven-plugin打出來的包是一個fat jar,即將所有的依賴全部打進了jar包當(dāng)中婉商,這樣的jar包體積很大晶通,每次更新系統(tǒng)的時候都需要完整替換整個jar包(本地還好,如果是云服務(wù)器华望,網(wǎng)速慢了每次上傳文件都想砸電腦π__π)蕊蝗。此外,系統(tǒng)切換環(huán)境時赖舟,也同時需要切換配置參數(shù)蓬戚,雖然可以使用配置中心或者利用命令行參數(shù)修改配置,但有時候也免不了直接需要修改配置文件宾抓,這樣的話就有必要將配置文件從jar包中分離出來子漩,單獨存放。
Spring社區(qū)大概也考慮到了部分開發(fā)者有這樣的需求石洗,所以提供了spring-boot-thin-launcher這個插件用來將項目的依賴和配置從jar包中分離出去幢泼。這個插件雖然是放到spring-projects-experimental(意思就是實驗性質(zhì)的項目)當(dāng)中的,但從我使用的經(jīng)驗來看應(yīng)該是比較穩(wěn)定的讲衫,能夠滿足絕大部分場景的需求旭绒。廢話少說,還是先上代碼吧:
<build>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>${java.version}</source>
<target>${java.version}</target>
<encoding>UTF-8</encoding>
</configuration>
</plugin>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>${spring-boot.version}</version>
<configuration>
<!--項目的執(zhí)行入口-->
<mainClass>com.example.Application</mainClass>
</configuration>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
<dependencies>
<dependency>
<groupId>org.springframework.boot.experimental</groupId>
<artifactId>spring-boot-thin-layout</artifactId>
<version>1.0.12.RELEASE</version>
</dependency>
</dependencies>
</plugin>
<plugin>
<groupId>org.springframework.boot.experimental</groupId>
<artifactId>spring-boot-thin-maven-plugin</artifactId>
<version>1.0.12.RELEASE</version>
<executions>
<execution>
<!--在編譯時下載依賴包 -->
<id>resolve</id>
<goals>
<goal>resolve</goal>
</goals>
<inherited>false</inherited>
</execution>
</executions>
</plugin>
<!--移動配置文件到外部文件夾-->
<plugin>
<artifactId>maven-antrun-plugin</artifactId>
<executions>
<execution>
<phase>compile</phase>
<goals>
<goal>run</goal>
</goals>
<configuration>
<tasks>
<move file="${project.build.directory}/classes/application.yml" todir="${project.build.directory}/thin/root/config"/>
<copy todir="${project.build.directory}/thin/root/">
<fileset dir="${basedir}/bin"/>
</copy>
</tasks>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
在maven-antrun-plugin中可以根據(jù)需要定義一些額外的任務(wù)焦人,比如移動其它的配置文件挥吵,或者將外部的一些文件加入到項目目錄中(比如執(zhí)行腳本等)。
這樣配置后花椭,打包以后的target目錄大概是這樣的:
其中thin目錄下面就可以作為整個項目的部署目錄了忽匈,config是配置文件的存放目錄,respository下面是所有的依賴包矿辽,backend-2.0.jar中僅包含了項目自身的資源丹允,體積比之前小太多了。
運行原理
那為什么執(zhí)行時袋倔,系統(tǒng)知道自動去外部加載依賴和配置呢雕蔽?這個問題需要首先了解打包后的結(jié)構(gòu),把backend-2.0.jar解壓之后宾娜,會發(fā)現(xiàn)除了我們自己的類和資源以外批狐,還多了一個類:
org.springframework.boot.loader.wrapper.ThinJarWrapper,其實我們在運行項目時,這個類才是真正的項目入口嚣艇,它會在默認(rèn)位置查找項目相關(guān)的依賴承冰,如果沒有找到,甚至還會從指定的maven倉庫中直接下載食零,所以啟動時系統(tǒng)能夠識別到外部的依賴包困乒。至于外部配置,是因為Spring Boot框架在讀取配置文件時贰谣,會默認(rèn)讀取幾個目錄下的配置文件娜搂,其中優(yōu)先級最高的就是當(dāng)前目錄下的config目錄(所以config目錄的名字不能改成其它的)。
執(zhí)行jar包的時候需要注意吱抚,要額外添加一個參數(shù)來指定依賴包所在的倉庫位置(在我們的配置中就在jar包的當(dāng)前文件夾)涌攻,例如:-Dthin.root=. 默認(rèn)的位置是${user.home}/.m2,如果倉庫中沒有需要的依賴频伤,啟動jar包時還會自動連接遠程倉庫進行下載,導(dǎo)致啟動時間非常長芝此,這一點需要注意憋肖。spring-boot-thin-launcher還有很多可配置的參數(shù),具體可以到 官網(wǎng) 上自行查看婚苹。
另外附上一個通用的spring-boot-thin-launcher打包文件的啟動腳本:
#!/bin/bash
#這里指定需要運行的jar包的名字
APP_NAME="your-jar-name.jar"
JVM_ARGS="-Xms512M -Xmx2048M -XX:+UseG1GC -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/opt/log/heap.hprof"
THIN_ARGS="-Dthin.root=. -Dthin.offline=true"
#使用說明岸更,用來提示輸入?yún)?shù)
usage() {
echo "Usage: sh 執(zhí)行腳本.sh [start|stop|restart|status|run]"
exit 1
}
#檢查程序是否在運行
is_exist(){
pid=`ps -ef|grep $APP_NAME|grep -v grep|awk '{print $2}' `
#如果不存在返回1,存在返回0
if [ -z "${pid}" ]; then
return 1
else
return 0
fi
}
#后臺啟動
start(){
is_exist
if [ $? -eq "0" ]; then
echo "${APP_NAME} is already running. pid=${pid} ."
else
echo "${APP_NAME} running with args: nohup java $THIN_ARGS -jar $JVM_ARGS $APP_NAME "
nohup java $THIN_ARGS -jar $JVM_ARGS $APP_NAME >> catalina.out 2>&1 &
fi
}
#前臺啟動
run(){
is_exist
if [ $? -eq "0" ]; then
echo "${APP_NAME} is already running. pid=${pid} ."
else
echo "${APP_NAME} running with args: java $THIN_ARGS -jar $JVM_ARGS $APP_NAME"
java $THIN_ARGS -jar $JVM_ARGS $APP_NAME
fi
}
#停止方法
stop(){
is_exist
if [ $? -eq "0" ]; then
kill -9 $pid
else
echo "${APP_NAME} is not running"
fi
}
#輸出運行狀態(tài)
status(){
is_exist
if [ $? -eq "0" ]; then
echo "${APP_NAME} is running. Pid is ${pid}"
else
echo "${APP_NAME} is NOT running."
fi
}
#重啟
restart(){
stop
start
}
#根據(jù)輸入?yún)?shù)膊升,選擇執(zhí)行對應(yīng)方法怎炊,不輸入則執(zhí)行使用說明
case "$1" in
"start")
start
;;
"run")
run
;;
"stop")
stop
;;
"status")
status
;;
"restart")
restart
;;
*)
usage
;;
esac