java线程池的基本使用

四种线程池

Executors中提供了四种线程池:

  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

控制一组任务

ExecutorService提供了invokeAnyinvokeAll方法,它们是批量执行的最常用形式,它们执行任务collection,然后等待至少一个,或全部任务完成

/**
执行给定的任务,当所有任务完成时,返回保持任务状态和结果的 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方法则返回所有任务的结果,可以这样来对结果进行处理:

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

这样处理的一个弊端是,如果第一个任务花费了很长时间,则不得不等待。在某些情况下,可能只需要一个任务出了结果就可以中止所有任务,这样就得不偿失。将结果按照可获得的顺序保存起来可能更好,这时需要用到ExecutorCompletionService来进行排列:

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
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java语言中的线程池可以通过`java.util.concurrent.Executors`类来创建和管理。线程池可以提高程序的性能和资源利用率,通过重用线程避免了线程的频繁创建和销毁。 下面是一个简单的示例来演示线程池使用: ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class ThreadPoolExample { public static void main(String[] args) { // 创建一个固定大小的线程池,大小为5 ExecutorService executor = Executors.newFixedThreadPool(5); for (int i = 0; i < 10; i++) { final int taskId = i; executor.execute(new Runnable() { public void run() { System.out.println("Task " + taskId + " is running."); try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("Task " + taskId + " is completed."); } }); } // 关闭线程池 executor.shutdown(); } } ``` 在上述示例中,我们使用`Executors.newFixedThreadPool()`方法创建了一个固定大小为5的线程池。然后,我们使用`execute()`方法向线程池提交了10个任务(通过`Runnable`接口的实现类),每个任务会打印出任务编号,并休眠2秒钟来模拟任务执行时间。最后,我们调用`shutdown()`方法关闭线程池。 这只是线程池基本使用方法,Java中还提供了其他类型的线程池,如`CachedThreadPool`和`ScheduledThreadPool`,以满足不同的需求。你可以根据具体场景选择合适的线程池类型。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值