`
tuhaitao
  • 浏览: 380831 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

tomcat关不掉的原因

 
阅读更多

tomcat/bin 目录下的catalina.sh是比较常用的shell

 

 

#启动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混乱,不知所措,ps 一看,大吃一惊,而后每次都用kill -9 <pid> 才放心。

 

 

其实不用那样,一般关闭不了的情况,是由于程序员自己在tomcat中开启了新的线程,而且未设置成daemon,造成的主线程不能退出.

 

怎么发现工程里到底哪里开启的新的非守护线程呢,其实jdk提供了jstack工具,可以帮助我们分析

 

查看方法很简单

 

$JAVA_HOME/bin/jstack  <pid>

 

pid是指进程ID, 用ps -ef|grep tomcat 就可以查看到:

 

 

[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查看:

 

[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的,所以是这样:

 

"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了 :)


2
0
分享到:
评论
2 楼 tuhaitao 2011-07-17  
longhua828 写道
谢谢分享,受教了!

嘿嘿
1 楼 longhua828 2011-07-13  
谢谢分享,受教了!

相关推荐

    关于Tomcat的AJP端口禁用.docx

    在禁用AJP端口之前,确保前端确实不依赖Apache服务,且没有其他系统或组件需要通过AJP与Tomcat交互。同时,如果禁用AJP后出现任何问题,可以回滚到备份的`server.xml.bak`文件,恢复原来的状态。 总之,禁用Tomcat...

    tomcat内存溢出问题解决经历

    前一段时间提交了一个产品版本给测试人员测试,测试结果简直出人意料! 测试一段时间后页面就卡死了,当时根据这个现象下意识的怀疑是...那就先解决下tomcat关不掉的问题吧,百度…检查代码…几十分钟后找到了,在tomc

    linux文件操作,linux下关闭tomcat失效的处理方法,linux下压缩文件

    5. 杀掉挂起进程:有时Tomcat服务可能因某些原因无法正常关闭,可以使用`pkill`或`killall`命令强制结束相关进程。 6. 检查依赖:确保所有Tomcat依赖的服务和库都已正确安装和配置,如Java环境。 三、Linux下压缩...

    解决Tomcat使用shutdown.bat关闭会将其他Tomcat关掉的问题

    本文将深入探讨这个问题的原因,并提供一个简单的解决方案。 首先,我们需要理解`shutdown.bat`脚本的工作原理。`shutdown.bat`是Tomcat提供的一个批处理文件,用于关闭正在运行的Tomcat服务。它会检查环境变量`...

    errors code

    在IT领域,特别是软件开发与部署中,“错误代码”(Errors Code)是常见的技术话题,它们对于诊断系统问题、理解程序运行状态具有至关重要的作用。本文将深入解析一个与Java Web应用开发相关的错误代码示例,该示例...

    SpringBoot之logback-spring.xml不生效的解决方法

    在Spring Boot应用中,日志配置是至关重要的,因为它负责记录应用程序的运行状态和错误信息。当使用logback作为日志框架时,通常我们会将配置文件放在`src/main/resources`目录下,命名为`logback-spring.xml`,这是...

    Nginx安装包

    再补充个上次漏掉的,这时我们可以关闭其中一个服务器,我关了tomcat1,再多次刷新页面,会发现接下来出现的都是tomcat2的页面,这时必然的,但是时而快时而慢。这其中原因是当nginx将请求转发到tomcat2时,服务器...

    大厂JAVA面试题库大全.pdf

    熟悉每一块内存的作用以及它们之间的关系,对于理解Java内存模型至关重要。 - **垃圾回收机制**:包括GC的基本原理、常见的GC算法(如复制算法、标记-整理算法)以及GC的触发条件等。 - **JVM参数调优**:掌握如何...

Global site tag (gtag.js) - Google Analytics