如何从容器中删除死线程

时间:2012-05-23 16:33:02

标签: java multithreading

我将线程存储在容器中。随着时间的推移,其中一些线程将会运行,其中一些将会死亡。我想要实现的是:自动(或定期)从容器中删除死(停止)线程。 这样做的最佳方式是什么?

编辑:我将我的主题存储在一个简单的链接列表中:

LinkedList<ServerThread> threadPool = new LinkedList<ServerThread>();

这个容器必须是动态的,因为随着时间的推移,我必须添加(并明显删除)线程。

EDIT2:这就是我目前管理线程的方式。正如你所看到的,我等待传入的连接,我不知道它何时会到达,但是当它到达时,我必须在一个新的线程中处理它。

while (!interrupted()) {
            try {
                Socket clientSocket = serverSocket.accept();
                if (portNumber == Server.SMTP_PORT_NUMBER) {
                    threadPool.add(new SMTPThread(clientSocket, db));
                    threadPool.getLast().setName("SMTP Thread " + ++threadCounter);
                } else {
                    threadPool.add(new POP3Thread(clientSocket, db));
                    threadPool.getLast().setName("POP3 Thread " + ++threadCounter);
                }

                threadPool.get(threadPool.size() - 1).start();

            } catch (SocketTimeoutException e) {
            } catch (IOException ioe) {
            }
        }

2 个答案:

答案 0 :(得分:5)

除非有特定原因,否则不应维护自己的线程列表。我建议使用自Java 5以来可用的优秀ExcecutorService类。如下所示:

// create a thread pool with 10 workers
ExecutorService threadPool = Executors.newFixedThreadPool(10);
// or you can create an open-ended thread pool
// ExecutorService threadPool = Executors.newCachedThreadPool();
for (Job job : jobsToDo) {
    threadPool.submit(new MyJobProcessor(job));
}
// once we have submitted all jobs to the thread pool, it should be shutdown
threadPool.shutdown();
...
public class MyJobProcessor implements Runnable {
    private Job job;
    public MyJobProcessor(Job job) {
        this.job = job;
    }
    public void run() {
        // process the job
    }
}

线程池将负责维护池中正在运行的线程。线程将在下一个作业中重复使用,并且当池中没有其他作业且已关闭时将关闭。你自己不需要收获任何死线。


修改

就您发布的代码而言,要从池中删除已完成的线程,您应该像以下一样运行它们:

 Iterator<ServerThread> iterator = threadPool.iterator();
 while (iterator.hasNext()) {
     ServerThread thread = iterator.next();
     if (!thread.isAlive()) {
        // remove it from the linked list
        iterator.remove();
     }
 }

每次向池中添加新线程时,之后都会执行此操作。另外,请记住,LinkedList无法进行get(#)方法调用。我建议你调整你的代码:

            Socket clientSocket = serverSocket.accept();
            ServerThread serverThread;
            if (portNumber == Server.SMTP_PORT_NUMBER) {
                serverThread = new SMTPThread(clientSocket, db);
                serverThread.setName("SMTP Thread " + ++threadCounter);
            } else {
                serverThread = new POP3Thread(clientSocket, db);
                serverThread.setName("POP3 Thread " + ++threadCounter);
            }
            serverThread.start();
            threadPool.put(serverThread);

答案 1 :(得分:1)

您可以使用Timer来安排将在列表中运行的定期TimerTask并删除死线程。这是一幅草图:

static final List<Thread> threads = new LinkedList<Thread>();
static final Timer timer = new Timer(true);
public static void main(String[] args) {
  timer.scheduleAtFixedRate(new TimerTask() {public void run() {
    for (Iterator<Thread> it = threads.iterator(); it.hasNext();)
      if (!it.next().isAlive()) it.remove();
  }}, 0, 10000);
}