实战Concurrent(1)

本文介绍Java多线程编程的基础知识及高级应用,重点讲解如何使用java.util.concurrent包中的工具类,如ExecutorService来简化多线程编程。通过具体案例演示如何创建线程池、执行任务并获取结果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

来自http://www.iteye.com/topic/363625

 

编写多线程的程序一直都是一件比较麻烦的事情,要考虑很多事情,处理不好还会出很多意想不到的麻烦。加上现在很多开发者接触到的项目都是打着企业级旗号的B/S项目,大多数人都很少涉及多线程,这又为本文的主角增加了一份神秘感。

 

讲到Java多线程,大多数人脑海中跳出来的是Thread、Runnable、synchronized……这些是最基本的东西,虽然已经足够强大,但想要用好还真不容易。从JDK 1.5开始,增加了java.util.concurrent包,它的引入大大简化了多线程程序的开发(要感谢一下大牛Doug Lee)。

 

java.util.concurrent包分成了三个部分,分别是java.util.concurrent、java.util.concurrent.atomic和java.util.concurrent.lock。内容涵盖了并发集合类、线程池机制、同步互斥机制、线程安全的变量更新工具类、锁等等常用工具。

 

为了便于理解,本文使用一个例子来做说明,交代一下它的场景:

假设要对一套10个节点组成的环境进行检查,这个环境有两个入口点,通过节点间的依赖关系可以遍历到整个环境。依赖关系可以构成一张有向图,可能存在环。为了提高检查的效率,考虑使用多线程。

 

1、Executors

通过这个类能够获得多种线程池的实例,例如可以调用newSingleThreadExecutor()获得单线程的ExecutorService,调用newFixedThreadPool()获得固定大小线程池的ExecutorService。拿到ExecutorService可以做的事情就比较多了,最简单的是用它来执行Runnable对象,也可以执行一些实现了Callable<T>的对象。用Thread的start()方法没有返回值,如果该线程执行的方法有返回值那用ExecutorService就再好不过了,可以选择submit()、invokeAll()或者invokeAny(),根据具体情况选择合适的方法即可。

Java代码  
  1. package  service;  
  2.   
  3. import  java.util.ArrayList;  
  4. import  java.util.List;  
  5. import  java.util.concurrent.ExecutionException;  
  6. import  java.util.concurrent.ExecutorService;  
  7. import  java.util.concurrent.Executors;  
  8. import  java.util.concurrent.Future;  
  9. import  java.util.concurrent.TimeUnit;  
  10.   
  11. /**  
  12.  * 线程池服务类  
  13.  *   
  14.  * @author DigitalSonic  
  15.  */   
  16. public   class  ThreadPoolService {  
  17.     /**  
  18.      * 默认线程池大小  
  19.      */   
  20.     public   static   final   int   DEFAULT_POOL_SIZE    =  5 ;  
  21.   
  22.     /**  
  23.      * 默认一个任务的超时时间,单位为毫秒  
  24.      */   
  25.     public   static   final   long  DEFAULT_TASK_TIMEOUT =  1000 ;  
  26.   
  27.     private   int               poolSize             = DEFAULT_POOL_SIZE;  
  28.     private  ExecutorService  executorService;  
  29.   
  30.     /**  
  31.      * 根据给定大小创建线程池  
  32.      */   
  33.     public  ThreadPoolService( int  poolSize) {  
  34.         setPoolSize(poolSize);  
  35.     }  
  36.   
  37.     /**  
  38.      * 使用线程池中的线程来执行任务  
  39.      */   
  40.     public   void  execute(Runnable task) {  
  41.         executorService.execute(task);  
  42.     }  
  43.   
  44.     /**  
  45.      * 在线程池中执行所有给定的任务并取回运行结果,使用默认超时时间  
  46.      *   
  47.      * @see #invokeAll(List, long)  
  48.      */   
  49.     public  List<Node> invokeAll(List<ValidationTask> tasks) {  
  50.         return  invokeAll(tasks, DEFAULT_TASK_TIMEOUT * tasks.size());  
  51.     }  
  52.   
  53.     /**  
  54.      * 在线程池中执行所有给定的任务并取回运行结果  
  55.      *   
  56.      * @param timeout 以毫秒为单位的超时时间,小于0表示不设定超时  
  57.      * @see java.util.concurrent.ExecutorService#invokeAll(java.util.Collection)  
  58.      */   
  59.     public  List<Node> invokeAll(List<ValidationTask> tasks,  long  timeout) {  
  60.         List<Node> nodes = new  ArrayList<Node>(tasks.size());  
  61.         try  {  
  62.             List<Future<Node>> futures = null ;  
  63.             if  (timeout <  0 ) {  
  64.                 futures = executorService.invokeAll(tasks);  
  65.             } else  {  
  66.                 futures = executorService.invokeAll(tasks, timeout, TimeUnit.MILLISECONDS);  
  67.             }  
  68.             for  (Future<Node> future : futures) {  
  69.                 try  {  
  70.                     nodes.add(future.get());  
  71.                 } catch  (ExecutionException e) {  
  72.                     e.printStackTrace();  
  73.                 }  
  74.             }  
  75.         } catch  (InterruptedException e) {  
  76.             e.printStackTrace();  
  77.         }  
  78.         return  nodes;  
  79.     }  
  80.   
  81.     /**  
  82.      * 关闭当前ExecutorService  
  83.      *   
  84.      * @param timeout 以毫秒为单位的超时时间  
  85.      */   
  86.     public   void  destoryExecutorService( long  timeout) {  
  87.         if  (executorService !=  null  && !executorService.isShutdown()) {  
  88.             try  {  
  89.                 executorService.awaitTermination(timeout, TimeUnit.MILLISECONDS);  
  90.             } catch  (InterruptedException e) {  
  91.                 e.printStackTrace();  
  92.             }  
  93.             executorService.shutdown();  
  94.         }  
  95.     }  
  96.   
  97.     /**  
  98.      * 关闭当前ExecutorService,随后根据poolSize创建新的ExecutorService  
  99.      */   
  100.     public   void  createExecutorService() {  
  101.         destoryExecutorService(1000 );  
  102.         executorService = Executors.newFixedThreadPool(poolSize);  
  103.     }  
  104.   
  105.     /**  
  106.      * 调整线程池大小  
  107.      * @see #createExecutorService()  
  108.      */   
  109.     public   void  setPoolSize( int  poolSize) {  
  110.         this .poolSize = poolSize;  
  111.         createExecutorService();  
  112.     }  
  113. }  
 

这里要额外说明一下invokeAll()和invokeAny()方法。前者会执行给定的所有Callable<T>对象,等所有任务完成后返回一个包含了执行结果的List<Future<T>>,每个Future.isDone()都是true,可以用Future.get()拿到结果;后者只要完成了列表中的任意一个任务就立刻返回,返回值就是执行结果。

还有一个比较诡异的地方
本代码是在JDK 1.6下编译测试的,如果在JDK 1.5下测试,很可能在invokeAll和invokeAny的地方出错。明明ValidationTask实现了 Callable<Node>,可是它死活不认,类型不匹配,这时可以将参数声明由List<ValidationTask>改为 List<Callable<Node>>。
造成这个问题的主要原因是两个版本中invokeAll和invokeAny的方法签名不同,1.6里是invokeAll(Collection<? extends Callable<T>> tasks),而1.5里是invokeAll(Collection<Callable<T>> tasks)。网上也有人遇到类似的问题( invokeAll() is not willing to acept a Collection<Callable<T>> )。

 

和其他资源一样,线程池在使用完毕后也需要释放,用shutdown()方法可以关闭线程池,如果当时池里还有没有被执行的任务,它会等待任务执行完毕,在等待期间试图进入线程池的任务将被拒绝。也可以用shutdownNow()来关闭线程池,它会立刻关闭线程池,没有执行的任务作为返回值返回。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值