溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Tomcat怎么做到一鍵式啟停

發(fā)布時間:2021-12-31 16:33:10 來源:億速云 閱讀:198 作者:iii 欄目:服務(wù)器

這篇文章主要講解了“Tomcat怎么做到一鍵式啟?!保闹械闹v解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Tomcat怎么做到一鍵式啟?!卑?!

腳本分析

start.sh和start.bat里面的內(nèi)容相同,所以這里就主要分析start.sh的內(nèi)容了。

os400=false case "`uname`" in OS400*) os400=true;; esac # resolve links - $0 may be a softlink # PRG是腳本路徑,如果當(dāng)前腳本文件為軟連接,則會解析出PRG真正文件所在的路徑 PRG="$0" while [ -h "$PRG" ] ; do # 判斷是否為軟連接  ls=`ls -ld "$PRG"` # 如果是軟連接,輸出中含有l(wèi)in -> source的字符串  link=`expr "$ls" : '.*-> \(.*\)$'` # 模式匹配出源文件的路徑  if expr "$link" : '/.*' > /dev/null; then # 正則匹配 /.* 這里expr會輸出匹配個數(shù),如果不為0,則說明$link包含目錄  PRG="$link"  else  PRG=`dirname "$PRG"`/"$link" # 當(dāng)不包含目錄,說明軟連接和源文件在同一目錄  fi done # 獲取腳本目錄路徑 PRGDIR=`dirname "$PRG"` EXECUTABLE=catalina.sh # Check that target executable exists if $os400; then  # -x will Only work on the os400 if the files are:  # 1. owned by the user  # 2. owned by the PRIMARY group of the user  # this will not work if the user belongs in secondary groups  eval else  if [ ! -x "$PRGDIR"/"$EXECUTABLE" ]; then  echo "Cannot find $PRGDIR/$EXECUTABLE"  echo "The file is absent or does not have execute permission"  echo "This file is needed to run this program"  exit 1  fi fi # 執(zhí)行catalina.sh的start命令 exec "$PRGDIR"/"$EXECUTABLE" start "$@"

其實上面簡單來說就做了兩件事

  1. 拿到腳本的真正路徑

  2. 執(zhí)行catalina.sh的start命令

而shutdown.sh和start.sh命令一樣,只不過后面是執(zhí)行catalina.sh的stop命令

catalina.sh腳本

腳本中重要的步驟有以下幾個

1.設(shè)置兩個重要的環(huán)境變量,CATALINA_HOME、CATALINA_BASE

PRGDIR=`dirname "$PRG"`     [ -z "$CATALINA_HOME" ] && CATALINA_HOME=`cd "$PRGDIR/.." >/dev/null; pwd`     [ -z "$CATALINA_BASE" ] && CATALINA_BASE="$CATALINA_HOME"

設(shè)置CLASSPATH變量,這里注意,默認(rèn)是沒有setenv.sh文件的,可以自己新建一個并添加參數(shù)

CLASSPATH=  if [ -r "$CATALINA_BASE/bin/setenv.sh" ]; then  . "$CATALINA_BASE/bin/setenv.sh"  elif [ -r "$CATALINA_HOME/bin/setenv.sh" ]; then  . "$CATALINA_HOME/bin/setenv.sh"  fi

將bootstrap.jar作為CLASSPATH變量傳進(jìn)去

if [ ! -z "$CLASSPATH" ] ; then      CLASSPATH="$CLASSPATH":     fi     CLASSPATH="$CLASSPATH""$CATALINA_HOME"/bin/bootstrap.jar     if [ -z "$CATALINA_OUT" ] ; then      CATALINA_OUT="$CATALINA_BASE"/logs/catalina.out     fi

執(zhí)行腳本參數(shù),執(zhí)行bootstrap.jar中的Bootstrap類中main方法,并傳入?yún)?shù)start

hift  eval exec "\"$_RUNJAVA\"" "\"$LOGGING_CONFIG\"" $LOGGING_MANAGER $JAVA_OPTS $CATALINA_OPTS \  -D$ENDORSED_PROP="\"$JAVA_ENDORSED_DIRS\"" \  -classpath "\"$CLASSPATH\"" \  -Djava.security.manager \  -Djava.security.policy=="\"$CATALINA_BASE/conf/catalina.policy\"" \  -Dcatalina.base="\"$CATALINA_BASE\"" \  -Dcatalina.home="\"$CATALINA_HOME\"" \  -Djava.io.tmpdir="\"$CATALINA_TMPDIR\"" \  org.apache.catalina.startup.Bootstrap "$@" start

在上面腳本中我們可以看出最后執(zhí)行的都是從Bootstrap的main方法作為入口的,所以我們打開Tomcat源碼進(jìn)去Bootstrap類中看它到底做了什么。

啟動類分析

作為Tomcat的入口類,我們先看看Bootstrap中做了什么。這里只貼出main方法中重要的代碼。

//初始化類加載器并且將Catalina文件加載進(jìn)內(nèi)存中 bootstrap.init(); String command = "start"; if (args.length > 0) {  command = args[args.length - 1]; } if (command.equals("startd")) {  args[args.length - 1] = "start";  //調(diào)用Catalina.java的load方法  daemon.load(args);  //調(diào)用Catalina.java的start  daemon.start(); } else if (command.equals("stopd")) {  args[args.length - 1] = "stop";  //調(diào)用Catalina.java的stop  daemon.stop(); } else if (command.equals("start")) {  daemon.setAwait(true);  daemon.load(args);  daemon.start();  if (null == daemon.getServer()) {  System.exit(1);  } } else if (command.equals("stop")) {  daemon.stopServer(args); } else if (command.equals("configtest")) {  daemon.load(args);  if (null == daemon.getServer()) {  System.exit(1);  }  System.exit(0); } else {  log.warn("Bootstrap: command \"" + command + "\" does not exist."); }

這里是根據(jù)腳本中傳入的不同命令,調(diào)用Catalina不同的方法。由于我們主要分析的Tomcat如何做到一鍵式啟停的,所以我們主要分析Catalina的start方法。

在Catalina的satrt方法中我們看到了這一句

getServer().start();

隨后經(jīng)過Debug都是經(jīng)過了Lifecycle的start方法,我們把Lifecycle的方法列出來

public interface Lifecycle {  public void addLifecycleListener(LifecycleListener listener);  public LifecycleListener[] findLifecycleListeners();  public void removeLifecycleListener(LifecycleListener listener);  public void init() throws LifecycleException;  public void start() throws LifecycleException;  public void stop() throws LifecycleException;  public void destroy() throws LifecycleException;  public LifecycleState getState();  public String getStateName();  public interface SingleUse {  } }

然后再看它的實現(xiàn)類,我們發(fā)現(xiàn)我們前面所講的整體架構(gòu)中的組件都實現(xiàn)了此類。而在它的子類LifecycleBase實現(xiàn)了start、init、stop等方法,并且里面都相應(yīng)調(diào)用了startInternal、initInternal、stopInternal方法,這里我們?nèi)绻麑τ谠O(shè)計模式了解的話,應(yīng)該會想到這里運(yùn)用了模板設(shè)計模式,抽象出所有子類的公有的代碼,然后重新定義一個內(nèi)部抽象方法,其子類實現(xiàn)自己的定制化的操作。

在Server.xml中我們發(fā)現(xiàn)第一個層級也是Server,然后Catalina的satrt方法中第一個啟動的也是Server。

Tomcat怎么做到一鍵式啟停

上面表示了Tomcat所有模塊的層級結(jié)構(gòu),只要是帶有層級的結(jié)構(gòu),我們應(yīng)該能夠立馬想到組合設(shè)計模式,從這個層級結(jié)構(gòu)中我們能夠得到模塊之間的關(guān)系,有大有小,有內(nèi)有外。

  • 有大有小:大組件管理小組件,例如Server管理Service,Service管理連接器和容器

  • 有內(nèi)有外:連接器控制對外的連接,而外層組件調(diào)用內(nèi)層組件完成業(yè)務(wù)功能。即請求處理的過程是由外層組件驅(qū)動的。

那么根據(jù)上面的兩條,我們知道,有小才有大,有內(nèi)才有外。這也就是整個層級的加載順序,先加載小組件再加載大組件,先加載內(nèi)層組件再加載外層組件。此時我們應(yīng)該就明白了Tomcat是如何做到一鍵式啟停的了。通過層級結(jié)構(gòu),加載的優(yōu)先級。層層迭代進(jìn)行啟動。而停止和啟動差不多。也是層層迭代進(jìn)行停止。

感謝各位的閱讀,以上就是“Tomcat怎么做到一鍵式啟?!钡膬?nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Tomcat怎么做到一鍵式啟停這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI