【转】在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的目的。但...
  • shine0181
  • shine0181
  • 2012年07月04日 23:22
  • 4873

Java守护线程与非守护线程

http://www.cnblogs.com/super-d2/p/3348183.html - 在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程)...
  • u012440687
  • u012440687
  • 2016年07月11日 10:01
  • 1075

JAVA守护线程 非守护线程

笔记: 第一篇转载写的比较好,将守护线程同linux的守护进程概念进行了对比。当非守护线程执行完jvm就退出,不管是否还有守护线程在执行。所以守护线程尽量不要执行逻辑代码,顶多执行一些可有可...
  • vincentff7zg
  • vincentff7zg
  • 2017年02月21日 10:35
  • 324

Java 多线程:守护线程和非守护线程

Java 多线程:守护线程和非守护线程
  • Silent_Paladin
  • Silent_Paladin
  • 2017年01月19日 22:00
  • 763

守护线程和非守护线程

1.java线程分为两类,守护线程和非守护线程 2.守护线程是指程序运行的时候在后台提供一种通用服务的线程。比如GC线程就是一个很称职的守护线程,这种线程并不是程序中不可或缺的部分。因此,但所有非守...
  • jiangtao7913
  • jiangtao7913
  • 2015年12月06日 16:30
  • 1084

Python3多线程--守护线程&非守护线程

1、说明:设置一个线程是守护线程,就说明这不是一个很重要的线程,对于这样的线程,只要主线程运行结束,就会直接退出。而如果一个线程不是守护线程的话,即使主线程运行结束也不会退出,而是等待所有的非守护线程...
  • gaitiangai
  • gaitiangai
  • 2017年02月27日 15:07
  • 385

Java的守护线程和非守护线程

原文地址:http://www.cnblogs.com/super-d2/p/3348183.html 最近重新研究Java基础知识,发现以前太多知识知识略略带过了,比较说Java的线程机制...
  • tanga842428
  • tanga842428
  • 2016年10月11日 23:36
  • 804

什么是java的守护线程

线程 是程序中的执行线程。Java 虚拟机允许应用程序并发地运行多个执行线程。每个线程都有一个优先级,高优先级线程的执行优先于低优先级线程。每个线程都可以或不可以标记为一个守护程序。当某个线程中运行的...
  • ljyy2006
  • ljyy2006
  • 2007年11月24日 21:49
  • 7091

java.util.concurrent 包中Executor与Executors的区别

1、Executor是一个接口 2、Executors是一个类
  • gjf281
  • gjf281
  • 2014年05月22日 12:39
  • 4632

eclipse部署非WEB项目到Tomcat

eclipse部署非WEB项目到Tomcat
  • AcceGrow
  • AcceGrow
  • 2015年06月01日 10:31
  • 2255
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:【转】在java web项目中慎用Executors以及非守护线程
举报原因:
原因补充:

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