tomcat classpath两个路径_死磕Tomcat系列(3)——Tomcat如何做到一键式启停的

点击上方“Java技术前线”,选择“置顶或者星标”

与你一起成长

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

脚本分析

start.shstart.bat里面的内容相同,所以这里就主要分析 start.sh的内容了。

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 "$@"

复制代码

其实上面简单来说就做了两件事

  1. 拿到脚本的真正路径

  2. 执行 catalina.sh的 start命令

shutdown.shstart.sh命令一样,只不过后面是执行 catalina.shstop命令

catalina.sh脚本

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

  1. 设置两个重要的环境变量, CATALINA_HOME、 CATALINA_BASE

PRGDIR=`dirname "$PRG"`

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

[ -z "$CATALINA_BASE" ] && CATALINA_BASE="$CATALINA_HOME"

复制代码

  1. 设置 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

复制代码

  1. 将 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

复制代码

  1. 执行脚本参数,执行 bootstrap.jar中的 Bootstrap类中 main方法,并传入参数 start

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

a54258d7d74f3062ceb954a2f6b6252d.png

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

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

  • 有内有外:连接器控制对外的连接,而外层组件调用内层组件完成业务功能。即请求处理的过程是由外层组件驱动的。

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

作者:不学无数的程序员

来源:https://juejin.im/post/5d1335996fb9a07ebb05437a

热门内容:

  • 888G面试资源分享
  • Mybatis教程1:MyBatis快速入门
  • MyBatis教程2:使用MyBatis对表执行CRUD操作
  • MyBatis教程3:优化MyBatis配置文件中的配置
  • MyBatis教程4:解决字段名与实体类属性名不相同的冲突
  • 2019年Java经典面试题汇总
  • Maven教程1:Maven入门
  • Maven教程2:Maven项目构建过程练习
  • Maven教程3:使用Maven构建项目
  • Maven教程4:Maven核心概念
  • Maven教程5: 聚合与继承
  • Maven教程6: Maven与Eclipse整合
  • Maven教程7:eclipse中使用Maven创建Web项目.md
  • Maven教程8: 使用Maven构建多模块项目
  • Maven教程9: 使用Nexus搭建Maven私服

a43194bcca088dde2c1937db6fc5140c.png

喜欢就点个"在看"呗^_^

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值