Servlet“已启动一个线程,但未能阻止它” - Tomcat中的内存泄漏

Apache Tomcat多次说:

Web 应用程序 [/MyServlet] 似乎启动了一个名为 [pool-61-thread-2] 的线程,但未能停止它。这很可能会造成内存泄漏。

这很危险吗?servlet 应该能够处理 10.000 个请求/天。完成后如何关闭线程?

class Worker {

        private final CountDownLatch startSignal;
        private final CountDownLatch doneSignal;
        private final int threadNumber;

        Worker(
                CountDownLatch startSignal,
                CountDownLatch doneSignal,
                int threadNumber
        ){

            this.startSignal = startSignal;
            this.doneSignal = doneSignal;
            this.threadNumber = threadNumber;

        }

        public String[][] getSomeStrArrArr() {

            String[][] isRs = new String[8][20];
            String[][] inRs = new String[8][20];
            String[][] iwRs = new String[8][20];

            try {

                startSignal.await();

                if (threadNumber == 1) {
                    // get String[][] result for thread number 1
                    isRs = getIS(erg1, erg2, request);

                }

                if (threadNumber == 2) {
                    // get String[][] result for thread number 2
                    inRs = getIN(search_plz, request);
                }

                if (threadNumber == 3) {
                    // get String[][] result for thread number 3
                    iwRs = getIW(erg1, erg2, request);
                }

                doneSignal.countDown();

            } catch (InterruptedException ex) {

                System.out.println(
                        "Thread number "+threadNumber+" has been interrupted."
                );

            }
            if (threadNumber == 1) {
                return isRs;
            }
            if (threadNumber == 2) {
                return inRs;
            }
            if (threadNumber == 3) {
                return iwRs;
            }
            return null;
        }


        public Callable<String[][]> getSomeCallableStrArrArr(){
            return new Callable<String[][]>() {
                public String[][] call() throws Exception {
                    return getSomeStrArrArr();
                }
            };
        }

    }

    ExecutorService pool = Executors.newFixedThreadPool(3);
    Set<Future<String[][]>> set = new HashSet<Future<String[][]>>();
    CountDownLatch startSignal = new CountDownLatch(1);
    CountDownLatch doneSignal = new CountDownLatch(3);
    for (int i=1;i<=3;i++) {
        Worker worker = new Worker(startSignal,doneSignal,i);
        Callable<String[][]> callable =
                worker.getSomeCallableStrArrArr();
        Future<String[][]> future = pool.submit(callable);
        set.add(future);
    }
    startSignal.countDown();
    try {
        doneSignal.await();

答案 1

是的,这是一个问题。如果您的代码启动非守护程序线程,则这些线程将继续工作,直到它们退出其 run 方法。即使其他所有内容都已完成,旧的JVM也会在这些线程继续运行时徘徊。如果您启动一个新实例,则可能会遇到旧线程仍在与新实例创建的线程一起工作的情况。

需要对任务进行设计,以便它们能够对中断做出反应(而不是吃掉异常并继续进行,这就是您的示例所显示的)。这意味着检查当前线程上的中断标志,并以有用的方式捕获中断的异常,允许任务中断其工作,并在需要时重置中断的标志。ExecutorService 实现具有一个 shutdownNow 方法,该方法将中断当前任务。

下面是如何使用中断停止线程的示例

确保执行器被关闭,你可以在ServletContextListener中处理这个问题


答案 2

推荐