死磕Tomcat系列(3)——Tomcat如何作到一键式启停的

死磕Tomcat系列(3)——Tomcat如何作到一键式启停的

在没有SpringBoot内嵌有Tomcat以前,咱们都是将项目打为War包放在Tomcat的webapp目录下面,而后若是是Linux系统,运行命令start.sh、若是是Windows系统,运行命令start.bat之后就能启动起来并访问到页面。若是是想要中止运行只须要运行shutdown.sh或者shutdown.bat就能将程序中止起来,那么Tomcat是如何作到只须要一个命令就将全部容器启动起来呢?html

脚本分析

start.sh start.bat里面的内容相同,因此这里就主要分析start.sh的内容了。java

os400=false
case "`uname`" in
OS400*) os400=true;;
esac

# resolve links - $0 may be a softlink
# PRG是脚本路径,若是当前脚本文件为软链接,则会解析出PRG真正文件所在的路径
PRG="$0"

while [ -h "$PRG" ] ; do # 判断是否为软链接
  ls=`ls -ld "$PRG"`   # 若是是软链接,输出中含有lin -> source的字符串
  link=`expr "$ls" : '.*-> \(.*\)$'` # 模式匹配出源文件的路径
  if expr "$link" : '/.*' > /dev/null; then # 正则匹配 /.* 这里expr会输出匹配个数,若是不为0,则说明$link包含目录
    PRG="$link"
  else
    PRG=`dirname "$PRG"`/"$link" # 当不包含目录,说明软链接和源文件在同一目录
  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

# 执行catalina.sh的start命令
exec "$PRGDIR"/"$EXECUTABLE" start "$@"

其实上面简单来讲就作了两件事linux

  1. 拿到脚本的真正路径
  2. 执行catalina.shstart命令

shutdown.shstart.sh命令同样,只不事后面是执行catalina.shstop命令web

catalina.sh脚本

脚本中重要的步骤有如下几个apache

  1. 设置两个重要的环境变量,CATALINA_HOMECATALINA_BASEbootstrap

    PRGDIR=`dirname "$PRG"`
    
    	[ -z "$CATALINA_HOME" ] && CATALINA_HOME=`cd "$PRGDIR/.." >/dev/null; pwd`
    
    	[ -z "$CATALINA_BASE" ] && CATALINA_BASE="$CATALINA_HOME"
  2. 设置CLASSPATH变量,这里注意,默认是没有setenv.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
  3. bootstrap.jar做为CLASSPATH变量传进去架构

    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
  4. 执行脚本参数,执行bootstrap.jar中的Bootstrap类中main方法,并传入参数startapp

    shift
    	    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

在上面脚本中咱们能够看出最后执行的都是从Bootstrapmain方法做为入口的,因此咱们打开Tomcat源码进去Bootstrap 类中看它到底作了什么。框架

启动类分析

做为Tomcat的入口类,咱们先看看Bootstrap中作了什么。这里只贴出main方法中重要的代码。

//初始化类加载器而且将Catalina文件加载进内存中
bootstrap.init();
String command = "start";
if (args.length > 0) {
    command = args[args.length - 1];
}

if (command.equals("startd")) {
    args[args.length - 1] = "start";
    //调用Catalina.java的load方法
    daemon.load(args);
    //调用Catalina.java的start
    daemon.start();
} else if (command.equals("stopd")) {
    args[args.length - 1] = "stop";
    //调用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.");
}

这里是根据脚本中传入的不一样命令,调用Catalina不一样的方法。因为咱们主要分析的Tomcat如何作到一键式启停的,因此咱们主要分析Catalinastart方法。

Catalinasatrt方法中咱们看到了这一句

getServer().start();

随后通过Debug都是通过了Lifecyclestart方法,咱们把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 {
    }
}

而后再看它的实现类,咱们发现咱们前面所讲的总体架构中的组件都实现了此类。而在它的子类LifecycleBase实现了startinitstop等方法,而且里面都相应调用了startInternalinitInternalstopInternal方法,这里咱们若是对于设计模式了解的话,应该会想到这里运用了模板设计模式,抽象出全部子类的公有的代码,而后从新定义一个内部抽象方法,其子类实现本身的定制化的操做。

Server.xml中咱们发现第一个层级也是Server,而后Catalinasatrt方法中第一个启动的也是Server

上面表示了Tomcat全部模块的层级结构,只要是带有层级的结构,咱们应该可以立马想到组合设计模式,从这个层级结构中咱们可以获得模块之间的关系,有大有小,有内有外

  • 有大有小:大组件管理小组件,例如Server管理Service,Service管理链接器和容器
  • 有内有外:链接器控制对外的链接,而外层组件调用内层组件完成业务功能。即请求处理的过程是由外层组件驱动的。

那么根据上面的两条,咱们知道,有小才有大,有内才有外。这也就是整个层级的加载顺序,先加载小组件再加载大组件,先加载内层组件再加载外层组件。此时咱们应该就明白了Tomcat是如何作到一键式启停的了。经过层级结构,加载的优先级。层层迭代进行启动。而中止和启动差很少。也是层层迭代进行中止。

往期文章

如何断点调试Tomcat源码

死磕Tomcat系列(1)——总体架构

死磕Tomcat系列(2)——EndPoint源码解析

一次奇怪的StackOverflowError问题查找之旅

徒手撸一个简单的RPC框架

徒手撸一个简单的RPC框架(2)——项目改造

参考文章

相关文章
相关标签/搜索