# 问题分析
在Linux环境下,使用shutdown.sh脚本关闭Tomcat,经常会出现无法正常关闭的情况,使用ps -ef命令查看发现仍然有tomcat的进程。
查看linux日志以及源代码,问题出在WebappClassLoaderBase中,此类为Context的类加载器父类,在容器关闭时,会调用其stop方法,stop方法会调用clearReferencesThreads方法
```java
private boolean clearReferencesStopThreads = false;
private void clearReferencesThreads() {
Thread[] threads = getThreads();
ListexecutorThreadsToStop = new ArrayList<>();
// Iterate over the set of threads
for (Thread thread : threads) {
if (thread != null) {
...省略其他代码...
// Don't try and stop the threads unless explicitly
// configured to do so
if (!clearReferencesStopThreads) {
continue;
}
...省略其他代码...
}
```
如果存在仍然运行的线程,根据代码默认的配置,则不会进行任何处理,所以tomcat的进程仍然会存在。
# 解决方案
解决这个问题的思路为,通过kill命令强制杀掉进程,本文提供两种方案。
### 1.自己编写shell脚本
其思路为先查询到tomcat的进程ID,然后杀掉进程,具体shell脚本如下:
```shell
#!/bin/bash
#TOMCAT_BASE为服务器上tomcat的bin目录路径
export TOMCAT_BASE=/usr/local/tomcat/7.0/bin
echo ' >>> deploy start'
#仍然先运行shutdown.sh脚本关闭tomcat
sh $TOMCAT_BASE/shutdown.sh
#查询tomcat的进程ID,并杀掉该进程
pidList=$(ps aux | grep $TOMCAT_BASE | grep -v grep | awk '{print $2}')
for pid in $pidList
do
kill -9 $pid
echo 'kill pid '$pid
done
#启动tomcat
sh $TOMCAT_BASE/startup.sh
echo ' >>> deploy end'
```
通过以上脚本,可以重启Tomcat。此方式的风险在于查找tomcat的进程ID是否准确,但可以一并杀掉因人为操作shutdown.sh而未关闭的进程。
### 2.Tomcat预留的窗口
在Tomcat的catalina.sh中预留了该问题的处理窗口,需进行以下配置:
1. 在bin目录中创建文件,文件名为CATALINA_PID
2. 在shutdown.sh中增加参数-force
```shell
exec "$PRGDIR"/"$EXECUTABLE" stop -force "$@"
```
3. 在catalina.sh中最前面增加以下代码,将变量$CATALINA_PID指向创建的CATALINA_PID文件,在Tomcat启动时,会将进程ID存储在这个文件中。
```shell
if [ -z "$CATALINA_PID" ]; then
CATALINA_PID=$PRGDIR/CATALINA_PID
fi
```
至此,运行shutdown.sh,当tomcat无法正常关闭时,会强制杀掉tomcat的进程。