java线程池的基本使用

四种线程池

Executors中提供了四种线程池:segmentfault

  1. newCachedThreadPool 可缓存线程池,对于每一个线程,若是有空闲线程可用,当即让它执行,若是没有,则建立一个新线程
  2. newFixedThreadPool 具备固定大小的线程池,若是任务数大于空闲的线程数,则把它们放进队列中等待
  3. newSingleThreadPool大小为1的线程池,任务一个接着一个完成
  4. newScheduledThreadPool 定长线程池,可控制线程最大并发数,支持定时及周期性任务执行,用来代替Timer

基本方法

在上文http://segmentfault.com/a/1190000003091174 中说到了callable不能直接被Thread运行,但却能被线程池运行,ExecutorService提供了几种方法运行一个任务:缓存

Future submit(Callable task);
Future submit(Runnable task, T result);
Future submit(Runnable task);并发

第一个方法能够直接提交一个Callable任务,返回一个包含结果的Future<T>,第二个方法会返回指定的result对象,第三个方法返回一个Future<?>,可使用这样的对象来调用isDone,cancel,isCancelled,可是在get的时候返回null。线程

此外,有两个经常使用的关闭线程池的方法:code

void shutdown();
List<Runnable> shutdownNow()对象

第一个方法将启动一次顺序关闭,有任务在执行,则等待执行完成,但不接受新的任务;
第二个方法将取消全部未开始的任务而且试图中断正在执行的任务,返回从未开始执行的任务的列表。没法保证可以中止正在处理的活动执行任务,可是会尽力尝试。例如,经过 Thread.interrupt() 来取消典型的实现,因此任何任务没法响应中断均可能永远没法终止。队列

控制一组任务

ExecutorService提供了invokeAnyinvokeAll方法,它们是批量执行的最经常使用形式,它们执行任务collection,而后等待至少一个,或所有任务完成get

/**
执行给定的任务,当全部任务完成时,返回保持任务状态和结果的 Future 列表。返回列表的全部元素的 Future.isDone() 为 true。
注意,能够正常地或经过抛出异常来终止已完成任务。
**/
<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks)
                          throws InterruptedException
/**
执行给定的任务,若是其中一个任务的结果。一旦正常或异常返回后,则取消还没有完成的任务。
**/                          
<T> T invokeAny(Collection<? extends Callable<T>> tasks)
            throws InterruptedException,
                   ExecutionException

invokeAny方法提交全部任务到一个Callable对象的集合中,而且返回某个已经完成了的任务的结果,返回的任务是不肯定的。invokeAll方法则返回全部任务的结果,能够这样来对结果进行处理:it

List<Callable<T>> tasks=...
List<Future<T>> results = executor.invokeAll(tasks);
for(Future<T> result : results){
    process(result.get());
}
...

这样处理的一个弊端是,若是第一个任务花费了很长时间,则不得不等待。在某些状况下,可能只须要一个任务出告终果就能够停止全部任务,这样就得不偿失。将结果按照可得到的顺序保存起来可能更好,这时须要用到ExecutorCompletionService来进行排列:io

ExecutorCompletionService service = new ExecutorCompletionService(executor);
for(Callable<T> task:tasks){
    service.submit(task);
}

for(int i = 0;i < task.size();i++){
    process(service.take().get());
}
...

其中,take()方法会移除下一个已经完成的结果(Future),若是没有可用结果则阻塞

使用小结

在使用线程池时,大多应该按照如下步骤:

  1. 调用Executors类中的静态方法newCachedThreadPoolnewFixedThreadPool建立线程池;
  2. 调用submit提交RunnableCallable任务;
  3. 若是想取消一个任务,或者提交了Callable对象,那就要保存好返回的Future对象;
  4. 当再也不提交任务时,调用shutdown
相关文章
相关标签/搜索