为什么 tomcat shutdown 后仍然有进程存在

tomcat shutdown 后经常会发现仍然有进程存在,其中tomcat/bin 目录下的catalina.sh是比较常用的shell,一般开启关闭tomcat操作如下:

1

2

3

4

#启动tomcat

./catalina.sh start

#关闭tomcat

./catalina.sh stop

搜索


  但是往往一个工程,开发一段时间后,会发现./catalina.sh stop关闭不了tomcat,而必须使用kill -9 <pid> 这样的强制命令去
  杀死tomcat,这么做当然可以,但是手法不是那么的优雅
  在tomat未被./catalina stop关闭的情况下,导致误以为tomcat已经关闭成功的哥们 会在更新完代码后,./catalina start一下,于是在服务器中就产生了2个tomcat的实例,log混乱,而后每次都用kill -9 <pid> 才放心。
  其实不用那样,一般关闭不了的情况,是由于程序员自己在tomcat中开启了新的线程,而且未设置成daemon,造成的主线程不能退出.
  怎么发现工程里到底哪里开启的新的非守护线程呢,其实jdk提供了jstack工具,可以帮助我们分析
  查看方法很简单
  $JAVA_HOME/bin/jstack  <pid>
  pid是指进程ID, 用ps -ef|grep tomcat 就可以查看到:
  

1

2

3

[root@localhost bin]# ps -ef|grep tomcat

  root      1513     1  2 23:41 pts/1    00:00:01 /usr/local/share/java/jdk1.6.0_25/bin/java -Djava.util.logging.config.file=/opt/apache-tomcat-6.0.32/conf/logging.properties -Djava.util.logging.manager=org.apache.juli.ClassLoaderLogManager -Djava.endorsed.dirs=/opt/apache-tomcat-6.0.32/endorsed -classpath /opt/apache-tomcat-6.0.32/bin/bootstrap.jar -Dcatalina.base=/opt/apache-tomcat-6.0.32 -Dcatalina.home=/opt/apache-tomcat-6.0.32 -Djava.io.tmpdir=/opt/apache-tomcat-6.0.32/temp org.apache.catalina.startup.Bootstrap start

  root      1544  1462  0 23:42 pts/1    00:00:00 grep --color=auto tomcat


  这里看到的进程ID是 1513
  调用jstack查看:
  

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

[root@localhost bin]# jstack 1513

  2011-07-12 23:44:00

  Full thread dump Java HotSpot(TM) Client VM (20.0-b11 mixed mode, sharing):

  "Attach Listener" daemon prio=10 tid=0xb41d7c00 nid=0x606 waiting on condition [0x00000000]

  java.lang.Thread.State: RUNNABLE

  "TP-Monitor" daemon prio=10 tid=0xb41d6400 nid=0x5fa in Object.wait() [0xb3e0b000]

  java.lang.Thread.State: TIMED_WAITING (on object monitor)

  at java.lang.Object.wait(Native Method)

  - waiting on <0x87143720> (a org.apache.tomcat.util.threads.ThreadPool$MonitorRunnable)

  at org.apache.tomcat.util.threads.ThreadPool$MonitorRunnable.run(ThreadPool.java:565)

  - locked <0x87143720> (a org.apache.tomcat.util.threads.ThreadPool$MonitorRunnable)

  at java.lang.Thread.run(Thread.java:662)

  "TP-Processor4" daemon prio=10 tid=0xb41d4c00 nid=0x5f9 runnable [0xb3e5c000]

  java.lang.Thread.State: RUNNABLE

  at java.net.PlainSocketImpl.socketAccept(Native Method)

  at java.net.PlainSocketImpl.accept(PlainSocketImpl.java:408)

  - locked <0x87124770> (a java.net.SocksSocketImpl)

  at java.net.ServerSocket.implAccept(ServerSocket.java:462)

  at java.net.ServerSocket.accept(ServerSocket.java:430)

  at org.apache.jk.common.ChannelSocket.accept(ChannelSocket.java:311)

  at org.apache.jk.common.ChannelSocket.acceptConnections(ChannelSocket.java:668)

  at org.apache.jk.common.ChannelSocket$SocketAcceptor.runIt(ChannelSocket.java:879)

  at org.apache.tomcat.util.threads.ThreadPool$ControlRunnable.run(ThreadPool.java:690)

  at java.lang.Thread.run(Thread.java:662)

  其中看到"Attach Listener" daemon prio=10 tid=0xb41d7c00 nid=0x606 waiting on condition [0x00000000]
  java.lang.Thread.State: RUNNABLE
  这行,最前变的"Attach Listener" 是线程名, 紧跟其后的 daemon是线程的守护状态,
  其中主线程不是daemon的,所以是这样:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

"main" prio=10 tid=0xb6d05000 nid=0x5ea runnable [0xb6ee9000]

java.lang.Thread.State: RUNNABLE

at java.net.PlainSocketImpl.socketAccept(Native Method)

at java.net.PlainSocketImpl.accept(PlainSocketImpl.java:408)

- locked <0x871644a8> (a java.net.SocksSocketImpl)

at java.net.ServerSocket.implAccept(ServerSocket.java:462)

at java.net.ServerSocket.accept(ServerSocket.java:430)

at org.apache.catalina.core.StandardServer.await(StandardServer.java:431)

at org.apache.catalina.startup.Catalina.await(Catalina.java:676)

at org.apache.catalina.startup.Catalina.start(Catalina.java:628)

at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)

at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)

at java.lang.reflect.Method.invoke(Method.java:597)

at org.apache.catalina.startup.Bootstrap.start(Bootstrap.java:289)

at org.apache.catalina.startup.Bootstrap.main(Bootstrap.java:414)


  在"main" 后没有daemon,看到这样的线程状态,顺藤摸瓜,找到对应new Thread的地方setDaemon(true)就可以,痛痛快快的./catalina stop了

转载于:https://my.oschina.net/u/2842177/blog/1552214

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Tomcat是一个开源的Java Web服务器,它是Apache软件基金会的一个项目。当我们停止Tomcat服务器时,可能会遇到Tomcat shutdown进程仍在运行的情况。 出现这种情况通常是由于某些正在运行的任务或线程没有正确关闭所致。在关闭Tomcat服务器时,Tomcat会尝试结束所有正在运行的任务和线程,然后关闭自身。但有时候,如果存在某些无法正常结束的任务或线程,就会导致Tomcat shutdown进程仍然存在。 要解决这个问题,我们可以尝试以下几种方法: 1. 查找并终止相关的进程:使用任务管理器或命令行工具,查找并手动终止与Tomcat相关的进程。通常可以通过进程名称或进程ID来识别。终止这些进程后,Tomcat shutdown进程应该会消失。 2. 强制关闭进程:如果无法通过常规方法终止Tomcat shutdown进程,我们可以尝试使用`kill`命令(在UNIX或Linux系统上)或`taskkill`命令(在Windows系统上)来强制关闭该进程。这需要在命令行中执行相应的命令,并提供Tomcat shutdown进程进程ID。 3. 检查Tomcat日志文件:查看Tomcat服务器的日志文件,例如catalina.out或catalina.log,以获取关于无法正常关闭的任务或线程的更多信息。根据日志中的报错信息,可以尝试修复相关的问题,并重新启动Tomcat服务器。 总的来说,当Tomcat shutdown进程仍然存在时,我们可以尝试通过手动终止相关进程、强制关闭进程以及检查Tomcat日志文件来解决该问题。这些方法将帮助我们确保Tomcat服务器完全关闭,并可以重新启动或进行其他需要操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值