【转】在java web项目中慎用Executors以及非守护线程

转载 2016年08月28日 23:44:55

 最近研究embeded tomcat,特别是关于tomcat启动和关闭的模块。通过查看相应的源代码, 我们知道tomcat的关闭是通过往相应的关闭端口发送指定的关闭指令来达到关闭tomcat的目的。但是有的时候,通过shutdown.bat或shutdown.sh却不能有效地关闭tomcat,网上也有很多人提出这个问题。通过相关资料,最后问题出现线程上。

     首先看java虚拟机退出的条件,如下所示:

1
2
a,调用了 Runtime 类的 exit 方法,并且安全管理器允许退出操作发生。
b,非守护线程的所有线程都已停止运行,无论是通过从对 run 方法的调用中返回,还是通过抛出一个传播到 run 方法之外的异常。

    如上所示,第一条是通过exit退出,第二条指出了一个正常程序退出的条件,就是所有的非守护线程都已经停止运行。我们看相应embed tomcat的启动代码,如下所示:

1
2
tomcat.start();
        tomcat.getServer().await();

    最后一条有效的运行命令即是await,通过调用shutdown命令时,这个await就会成功的返回。按照常理来说,整个程序即会成功的完成。但是程序有时候并没有成功的结束,原因就在于程序中还存在着非守护进程。
    对于tomcat来说,tomcat程序中开启的所有进程都是守护进程,所以tomcat自身可以保证程序的正常结束。当await结束时,tomcat所就正常的结束了,包括相应的监听端口等,都已经成功结束。然而,由于项目程序中仍然还有其它线程在运行,所以导致java虚拟机并没有成功的退出。

     在我们的项目中,很多时候都运用到了线程。比如,异步调用等。不过,幸运的是,这些线程往往都是守护线程,原因就在于tomcat在运行我们的项目时,对于每一个请求,tomcat是使用了守护线程来进行相应的请求调用,这个保证在以下代码:

AprEndpoint
01
02
03
04
05
06
07
08
09
10
// Start poller threads
            pollers = new Poller[pollerThreadCount];
            for (int i = 0; i < pollerThreadCount; i++) {
                pollers[i] = new Poller(false);
                pollers[i].init();
                Thread pollerThread = new Thread(pollers[i], getName() + "-Poller-" + i);
                pollerThread.setPriority(threadPriority);
                pollerThread.setDaemon(true);
                pollerThread.start();
            }

    所以,一般情况下,在我们的项目代码中使用new Thread建立的线程都是守护线程,原因就是新建线程默认上使用建立线程时的当前线程所处的守护状态。tomcat的请求处理线程为守护线程,所以我们一般情况下建立的线程也是守护线程。然而,Executors除外。

     使用Executors建立后台线程并执行一些多线程操作时,Executors会使用相对应的threadFactory来对runnable建立新的thread,所以使用默认的threadFactory时就会出问题。默认的ThreadFactory强制性的将新创建的线程设置为非守护状态,如下所示:

01
02
03
04
05
06
07
08
09
10
public Thread newThread(Runnable r) {
            Thread t = new Thread(group, r,
                                  namePrefix + threadNumber.getAndIncrement(),
                                  0);
            if (t.isDaemon())
                t.setDaemon(false);
            if (t.getPriority() != Thread.NORM_PRIORITY)
                t.setPriority(Thread.NORM_PRIORITY);
            return t;
        }

    所以,一般情况下,我们使用executors创建多线程时,就会使用默认的threadFactory(即调用只有一个参数的工厂方法),而创建出来的线程就是非守护的。而相应的程序就永远不会退出,如采用Executors创建定时调度任务时,这个调试任务永远不会退出。解决的办法就是重写相对应的threadFactory,如下所示:

1
2
3
4
5
6
7
new ThreadFactory() {
        public Thread newThread(Runnable r) {
            Thread s = Executors.defaultThreadFactory().newThread(r);
            s.setDaemon(true);
            return s;
        }
    }

    同理,对于java web项目中的线程程序,一定要记住将相应的线程标记为守护线程(尽管它默认就是守护的)。而对于使用Executors,一定要记住传递相应的threadFactory实现,以重写相应的newThread方法,将线程标记为守护线程。
    以上的结论对于普通的java项目同样有效,为了正常的结束相应的程序,一定要正确的使用相应的线程,以避免java程序不能退出的问题。

转载请标明出处:i flym
本文地址:https://www.iflym.com/index.php/code/dont-use-executors-and-nondaemon-thread-in-java-web.html

在java web项目中慎用Executors以及非守护线程

最近研究embeded tomcat,特别是关于tomcat启动和关闭的模块。通过查看相应的源代码, 我们知道tomcat的关闭是通过往相应的关闭端口发送指定的关闭指令来达到关闭tomcat的目的。但...

慎用守护线程Daemon

public class Test{ public static void main(String[] args){ int i=1; ...
  • SMCwwh
  • SMCwwh
  • 2011年09月07日 10:40
  • 1179

java的守护线程+log4日志记录

  • 2015年01月12日 09:51
  • 469KB
  • 下载

Java中守护线程的总结(转)

在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程)  用个比较通俗的比如,任何一个守护线程都是整个JVM中所有非守护线程的保姆: 只要当前JVM实例...

java daemon 守护线程实例

  • 2012年12月12日 15:25
  • 986B
  • 下载

一次web请求,进入tomcat和守护线程

l  一次web请求,进入tomcat 服务器处理请求的模式 1.       收到一个请求就处理,这个时候就不能处理新的请求,这种为阻塞 这个是单线程模型 2.       收到一个请求就新开...
  • muskter
  • muskter
  • 2016年12月21日 18:21
  • 1123

java多线程之守护线程(Daemon)

在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程)  用个比较通俗的比如,任何一个守护线程都是整个JVM中所有非守护线程的保姆: 只要当前J...

java多线程之守护线程(daemon thread)

转载请注明出处 http://blog.csdn.net/pony_maggie/article/details/42402657 作者:小马 daemon是相于user线程而言的,可以...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:【转】在java web项目中慎用Executors以及非守护线程
举报原因:
原因补充:

(最多只允许输入30个字)