`

【转】Java并发框架Executor学习笔记

阅读更多

Java SE5的java.util.concurrent包中的执行器(Executor)将为你管理Thread对象,从而简化了并发编程。Executor在客户端和执行任务之间提供了一个间接层,Executor代替客户端执行任务。Executor允许你管理异步任务的执行,而无须显式地管理线程的生命周期。Executor在Java SE5/6中时启动任务的优选方法。Executor引入了一些功能类来管理和使用线程Thread,其中包括线程池,Executor,Executors,ExecutorService,CompletionService,Future,Callable等


创建线程池

Executors类,提供了一系列工厂方法用于创先线程池,返回的线程池都实现了ExecutorService接口。

 

public static ExecutorService newFixedThreadPool(int nThreads)

创建固定数目线程的线程池。

public static ExecutorService newCachedThreadPool()

创建一个可缓存的线程池,调用execute 将重用以前构造的线程(如果线程可用)。如果现有线程没有可用的,则创建一个新线程并添加到池中。终止并从缓存中移除那些已有 60 秒钟未被使用的线程。

public static ExecutorService newSingleThreadExecutor()

创建一个单线程化的Executor。

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize)

创建一个支持定时及周期性的任务执行的线程池,多数情况下可用来替代Timer类。

 

见类图,接口Executor只有一个方法execute,接口ExecutorService扩展了Executor并添加了一些生命周期管理的方法,如shutdown、submit等。一个Executor的生命周期有三种状态,运行 ,关闭 ,终止。

 

Callable,Future用于返回结果

Future<V>代表一个异步执行的操作,通过get()方法可以获得操作的结果,如果异步操作还没有完成,则,get()会使当前线程阻塞。FutureTask<V>实现了Future<V>和Runable<V>。Callable代表一个有返回值得操作。

实例:并行计算求和

 

Java代码  收藏代码
  1. public class ConcurrentSum {  
  2.     private int coreCpuNum;  
  3.     private ExecutorService  executor;  
  4.     private List<FutureTask<Long>> tasks = new ArrayList<FutureTask<Long>>();  
  5.     public ConcurrentSum(){  
  6.         coreCpuNum = Runtime.getRuntime().availableProcessors();  
  7.         executor = Executors.newFixedThreadPool(coreCpuNum);  
  8.     }  
  9.     class SumCalculator implements Callable<Long>{  
  10.         int nums[];  
  11.         int start;  
  12.         int end;  
  13.         public SumCalculator(final int nums[],int start,int end){  
  14.             this.nums = nums;  
  15.             this.start = start;  
  16.             this.end = end;  
  17.         }  
  18.         @Override  
  19.         public Long call() throws Exception {  
  20.             long sum =0;  
  21.             for(int i=start;i<end;i++){  
  22.                 sum += nums[i];  
  23.             }  
  24.             return sum;  
  25.         }  
  26.     }  
  27.     public long sum(int[] nums){  
  28.         int start,end,increment;  
  29.         // 根据CPU核心个数拆分任务,创建FutureTask并提交到Executor   
  30.         for(int i=0;i<coreCpuNum;i++){  
  31.             increment = nums.length / coreCpuNum+1;  
  32.             start = i*increment;  
  33.             end = start+increment;  
  34.             if(end > nums.length){  
  35.                 end = nums.length;   
  36.             }  
  37.             SumCalculator calculator = new SumCalculator(nums, start, end);  
  38.             FutureTask<Long> task = new FutureTask<Long>(calculator);  
  39.             tasks.add(task);  
  40.             if(!executor.isShutdown()){  
  41.                 executor.submit(task);  
  42.             }  
  43.         }  
  44.         return getPartSum();  
  45.     }  
  46.     public long getPartSum(){  
  47.         long sum = 0;  
  48.         for(int i=0;i<tasks.size();i++){  
  49.             try {  
  50.                 sum += tasks.get(i).get();  
  51.             } catch (InterruptedException e) {  
  52.                 e.printStackTrace();  
  53.             } catch (ExecutionException e) {  
  54.                 e.printStackTrace();  
  55.             }  
  56.         }  
  57.         return sum;  
  58.     }  
  59.     public void close(){  
  60.         executor.shutdown();  
  61.     }  
  62.       
  63.     public static void main(String[] args) {  
  64.         int arr[] = new int[]{12233452617481011 };  
  65.         long sum = new ConcurrentSum().sum(arr);  
  66.         System.out.println("sum: " + sum);  
  67.     }  
  68. }  

 

CompletionService

在上述例子中,getResult()方法的实现过程中,迭代了FutureTask的数组,如果任务还没有完成则当前线程会阻塞,如果我们希望任意任务完成后就把其结果加到result中,而不用依次等待每个任务完成,可以使用CompletionService。

它与ExecutorService最主要的区别在于submit的task不一定是按照加入时的顺序完成的。CompletionService对ExecutorService进行了包装,内部维护一个保存Future对象的BlockingQueue。只有当这个Future对象状态是结束的时候,才会加入到这个Queue中,take()方法其实就是Producer-Consumer中的Consumer。它会从Queue中取出Future对象,如果Queue是空的,就会阻塞在那里,直到有完成的Future对象加入到Queue中。所以,先完成的必定先被取出。这样就减少了不必要的等待时间。

实例:并行计算求和

 

Java代码  收藏代码
  1. public class ConcurrentSum2 {  
  2.     private int coreCpuNum;  
  3.     private ExecutorService  executor;  
  4.     private CompletionService<Long> completionService;  
  5.       
  6.     public ConcurrentSum2(){  
  7.         //.....  
  8.     }  
  9.     class SumCalculator implements Callable<Long>{  
  10.         //.....  
  11.     }  
  12.     public long sum(int[] nums){  
  13.         int start,end,increment;  
  14.         // 根据CPU核心个数拆分任务,创建FutureTask并提交到Executor   
  15.         for(int i=0;i<coreCpuNum;i++){  
  16.             increment = nums.length / coreCpuNum+1;  
  17.             start = i*increment;  
  18.             end = start+increment;  
  19.             if(end > nums.length){  
  20.                 end = nums.length;   
  21.             }  
  22.             SumCalculator task = new SumCalculator(nums, start, end);  
  23.             if(!executor.isShutdown()){  
  24.                 completionService.submit(task);  
  25.             }  
  26.         }  
  27.         return getPartSum();  
  28.     }  
  29.     public long getPartSum(){  
  30.         long sum = 0;  
  31.         for(int i=0;i<coreCpuNum;i++){  
  32.             try {  
  33.                 sum += completionService.take().get();  
  34.             } catch (InterruptedException e) {  
  35.                 e.printStackTrace();  
  36.             } catch (ExecutionException e) {  
  37.                 e.printStackTrace();  
  38.             }  
  39.         }  
  40.         return sum;  
  41.     }  
  42.     public void close(){  
  43.         executor.shutdown();  
  44.     }  
  45. }  

 

转自:http://willsunforjava.iteye.com/blog/1631353

分享到:
评论

相关推荐

    JAVA并发编程实践-线程执行-学习笔记

    更推荐使用Executor框架,它是Java并发编程的一个核心组件。Executor框架提供了线程池服务,可以有效管理和复用线程,降低系统资源消耗,同时提供了一种标准化的方式来调度和执行任务。 Executor生命周期包括创建...

    JAVA课程学习笔记.doc

    本篇学习笔记将深入解析Java线程池的框架、结构、原理以及相关源码,帮助读者全面理解线程池的工作机制。 1. 线程池模块结构 线程池框架分为多层结构,其中包括核心实现类、辅助类和接口等组件。例如,`sun.nio.ch....

    Java+JDK+6学习笔记

    2. **并发编程**:`java.util.concurrent`包提供了线程安全的数据结构和并发工具类,如Executor框架,大大简化了多线程编程。 3. **XML处理**:JDK 6提供了SAX、DOM和StAX等多种XML解析方式,同时JAXB支持XML与Java...

    java线程学习笔记

    Java线程学习笔记涉及了Java多线程编程的多个关键知识点,本篇知识点整理将详细解释每个概念及其在Java中的实现方式。 基本知识部分包含了Java线程编程的基础内容,它们是并发编程的基石。 任务Runnable是一个接口...

    并发编程之一 日常学习笔记

    本篇笔记主要关注并发编程中的两个关键概念:CAS(Compare and Swap)原子操作和Java线程的深入理解。 首先,我们来详细探讨一下CAS(比较并交换)原子操作。CAS是一种无锁算法,它通过比较内存中的某个值与期望值...

    java学习笔记

    Java提供了Thread类和Runnable接口来创建和管理线程,同时还有synchronized关键字、Lock接口、Executor框架等工具来控制并发访问和同步。理解和运用这些机制,可以有效地利用多核处理器提高程序性能。 网络编程是...

    java学习笔记JDK6课件和课本代码

    8. **并发工具类(Concurrent Utilities)**:在`java.util.concurrent`包中添加了新的线程安全的数据结构和并发编程工具,如Executor框架,使多线程编程更加便捷和高效。 9. **JAXB 2.0**:Java Architecture for ...

    《IT学习资料3》-Java架构师学习.zip

    首先,我们来看"ConcurrentProgrammingStudyNotes",这可能是一份关于Java并发编程的学习笔记。并发编程涉及线程管理、同步机制、锁、并发容器、并发工具类等内容。Java提供了丰富的API来支持并发,如`java.util....

    java学习笔记.7z

    这份"java学习笔记.7z"压缩包中包含了作者在学习Java过程中整理的基础知识,旨在为初学者提供指导。以下是笔记中可能涵盖的重要知识点的详细解释: 1. **概述** - Java的历史:由Sun Microsystems开发,后被Oracle...

    java多线程学习笔记

    - **使用ExecutorService**:通过Executor框架创建线程池,如`ExecutorService executor = Executors.newFixedThreadPool(10);` 2. **线程的启动与生命周期** - **start()方法**:调用Thread对象的start()方法...

    Java基础(韩顺平版)笔记详

    ### Java基础(韩顺平版)笔记详 #### 一、Java语言概述与环境搭建 - **Java的历史与发展** - Java由Sun Microsystems公司在1995年发布,由James Gosling领导开发。 - 2009年,Oracle公司收购了Sun Microsystems...

    多线程-狂神说Java学习笔记

    本学习笔记将深入探讨Java多线程的相关知识,包括其原理、实现方式、同步机制以及常见问题。 ### 一、多线程的基本概念 多线程是指在一个程序中存在两个或更多的执行线程,这些线程共享同一内存空间,但各自拥有...

    【学习笔记】JUC基础基础程序

    本学习笔记将深入探讨JUC的基础知识,帮助你理解和掌握Java并发编程的核心概念。 在Java中,多线程是实现并发的主要方式。一个线程是程序执行的一个流,多线程则意味着一个程序内可以同时执行多个线程。Java提供了...

    java源码笔记

    以上就是Java源码笔记可能涉及的主要内容,通过深入学习这些知识点,开发者可以更好地理解和运用Java进行网络编程,提高软件开发的效率和质量。同时,对源码的深入理解也有助于解决实际问题,提升编程技能。

    日常工作学习笔记(Java)

    3. **线程池**:Executor框架提供了线程池,有效管理线程生命周期,减少创建和销毁线程的开销。 以上是Java编程的一些核心知识点,涵盖了从基础语法到高级特性的各个方面。深入理解和掌握这些内容对于日常的Java...

    JDK1.60学习笔记

    通过阅读《Java JDK 1.60学习笔记.chm》这样的资源,开发者可以深入学习这些主题,进一步提升自己的Java编程能力。这份文档可能涵盖了所有这些知识点的详细讲解,以及实例代码和最佳实践,对于初学者和经验丰富的...

    Java Concurrency In Practice Learning Note

    《Java并发编程实践》学习笔记 Java并发编程是Java开发者必须掌握的重要技能,尤其是在多核处理器和高并发应用中。本书《Java Concurrency In Practice》是Java并发编程的经典之作,由Brian Goetz、Tim Peierls、...

    JDK 6学习笔记——PPT简体版

    这份学习笔记将围绕JDK 6展开,涵盖其主要特性和基础Java知识。 1. **Java基础** - **语法**:Java是一种面向对象的编程语言,具有严格的类型检查和垃圾回收机制,保证了代码的稳定性和安全性。 - **类与对象**:...

    java高级.zip

    Java还提供了高级的并发工具,如Executor框架、Semaphore、CountDownLatch等,用于更有效地管理线程和同步。 3. **IO流**:Java的IO流允许程序进行输入和输出操作,无论是处理文件、网络数据还是内存中的数据。IO流...

    java多线程笔记全手打

    Java的Executor框架提供了线程池管理,通过`Executors`静态工厂方法创建线程池,如`newFixedThreadPool`创建固定大小的线程池,`newSingleThreadExecutor`创建只有一个工作线程的线程池。线程池能有效控制运行的线程...

Global site tag (gtag.js) - Google Analytics