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代表一个有返回值得操作。
实例:并行计算求和
- public class ConcurrentSum {
- private int coreCpuNum;
- private ExecutorService executor;
- private List<FutureTask<Long>> tasks = new ArrayList<FutureTask<Long>>();
- public ConcurrentSum(){
- coreCpuNum = Runtime.getRuntime().availableProcessors();
- executor = Executors.newFixedThreadPool(coreCpuNum);
- }
- class SumCalculator implements Callable<Long>{
- int nums[];
- int start;
- int end;
- public SumCalculator(final int nums[],int start,int end){
- this.nums = nums;
- this.start = start;
- this.end = end;
- }
- @Override
- public Long call() throws Exception {
- long sum =0;
- for(int i=start;i<end;i++){
- sum += nums[i];
- }
- return sum;
- }
- }
- public long sum(int[] nums){
- int start,end,increment;
- // 根据CPU核心个数拆分任务,创建FutureTask并提交到Executor
- for(int i=0;i<coreCpuNum;i++){
- increment = nums.length / coreCpuNum+1;
- start = i*increment;
- end = start+increment;
- if(end > nums.length){
- end = nums.length;
- }
- SumCalculator calculator = new SumCalculator(nums, start, end);
- FutureTask<Long> task = new FutureTask<Long>(calculator);
- tasks.add(task);
- if(!executor.isShutdown()){
- executor.submit(task);
- }
- }
- return getPartSum();
- }
- public long getPartSum(){
- long sum = 0;
- for(int i=0;i<tasks.size();i++){
- try {
- sum += tasks.get(i).get();
- } catch (InterruptedException e) {
- e.printStackTrace();
- } catch (ExecutionException e) {
- e.printStackTrace();
- }
- }
- return sum;
- }
- public void close(){
- executor.shutdown();
- }
- public static void main(String[] args) {
- int arr[] = new int[]{1, 22, 33, 4, 52, 61, 7, 48, 10, 11 };
- long sum = new ConcurrentSum().sum(arr);
- System.out.println("sum: " + sum);
- }
- }
CompletionService
在上述例子中,getResult()方法的实现过程中,迭代了FutureTask的数组,如果任务还没有完成则当前线程会阻塞,如果我们希望任意任务完成后就把其结果加到result中,而不用依次等待每个任务完成,可以使用CompletionService。
它与ExecutorService最主要的区别在于submit的task不一定是按照加入时的顺序完成的。CompletionService对ExecutorService进行了包装,内部维护一个保存Future对象的BlockingQueue。只有当这个Future对象状态是结束的时候,才会加入到这个Queue中,take()方法其实就是Producer-Consumer中的Consumer。它会从Queue中取出Future对象,如果Queue是空的,就会阻塞在那里,直到有完成的Future对象加入到Queue中。所以,先完成的必定先被取出。这样就减少了不必要的等待时间。
实例:并行计算求和
- public class ConcurrentSum2 {
- private int coreCpuNum;
- private ExecutorService executor;
- private CompletionService<Long> completionService;
- public ConcurrentSum2(){
- //.....
- }
- class SumCalculator implements Callable<Long>{
- //.....
- }
- public long sum(int[] nums){
- int start,end,increment;
- // 根据CPU核心个数拆分任务,创建FutureTask并提交到Executor
- for(int i=0;i<coreCpuNum;i++){
- increment = nums.length / coreCpuNum+1;
- start = i*increment;
- end = start+increment;
- if(end > nums.length){
- end = nums.length;
- }
- SumCalculator task = new SumCalculator(nums, start, end);
- if(!executor.isShutdown()){
- completionService.submit(task);
- }
- }
- return getPartSum();
- }
- public long getPartSum(){
- long sum = 0;
- for(int i=0;i<coreCpuNum;i++){
- try {
- sum += completionService.take().get();
- } catch (InterruptedException e) {
- e.printStackTrace();
- } catch (ExecutionException e) {
- e.printStackTrace();
- }
- }
- return sum;
- }
- public void close(){
- executor.shutdown();
- }
- }
转自:http://willsunforjava.iteye.com/blog/1631353
相关推荐
更推荐使用Executor框架,它是Java并发编程的一个核心组件。Executor框架提供了线程池服务,可以有效管理和复用线程,降低系统资源消耗,同时提供了一种标准化的方式来调度和执行任务。 Executor生命周期包括创建...
本篇学习笔记将深入解析Java线程池的框架、结构、原理以及相关源码,帮助读者全面理解线程池的工作机制。 1. 线程池模块结构 线程池框架分为多层结构,其中包括核心实现类、辅助类和接口等组件。例如,`sun.nio.ch....
2. **并发编程**:`java.util.concurrent`包提供了线程安全的数据结构和并发工具类,如Executor框架,大大简化了多线程编程。 3. **XML处理**:JDK 6提供了SAX、DOM和StAX等多种XML解析方式,同时JAXB支持XML与Java...
Java线程学习笔记涉及了Java多线程编程的多个关键知识点,本篇知识点整理将详细解释每个概念及其在Java中的实现方式。 基本知识部分包含了Java线程编程的基础内容,它们是并发编程的基石。 任务Runnable是一个接口...
本篇笔记主要关注并发编程中的两个关键概念:CAS(Compare and Swap)原子操作和Java线程的深入理解。 首先,我们来详细探讨一下CAS(比较并交换)原子操作。CAS是一种无锁算法,它通过比较内存中的某个值与期望值...
Java提供了Thread类和Runnable接口来创建和管理线程,同时还有synchronized关键字、Lock接口、Executor框架等工具来控制并发访问和同步。理解和运用这些机制,可以有效地利用多核处理器提高程序性能。 网络编程是...
8. **并发工具类(Concurrent Utilities)**:在`java.util.concurrent`包中添加了新的线程安全的数据结构和并发编程工具,如Executor框架,使多线程编程更加便捷和高效。 9. **JAXB 2.0**:Java Architecture for ...
首先,我们来看"ConcurrentProgrammingStudyNotes",这可能是一份关于Java并发编程的学习笔记。并发编程涉及线程管理、同步机制、锁、并发容器、并发工具类等内容。Java提供了丰富的API来支持并发,如`java.util....
这份"java学习笔记.7z"压缩包中包含了作者在学习Java过程中整理的基础知识,旨在为初学者提供指导。以下是笔记中可能涵盖的重要知识点的详细解释: 1. **概述** - Java的历史:由Sun Microsystems开发,后被Oracle...
- **使用ExecutorService**:通过Executor框架创建线程池,如`ExecutorService executor = Executors.newFixedThreadPool(10);` 2. **线程的启动与生命周期** - **start()方法**:调用Thread对象的start()方法...
### Java基础(韩顺平版)笔记详 #### 一、Java语言概述与环境搭建 - **Java的历史与发展** - Java由Sun Microsystems公司在1995年发布,由James Gosling领导开发。 - 2009年,Oracle公司收购了Sun Microsystems...
本学习笔记将深入探讨Java多线程的相关知识,包括其原理、实现方式、同步机制以及常见问题。 ### 一、多线程的基本概念 多线程是指在一个程序中存在两个或更多的执行线程,这些线程共享同一内存空间,但各自拥有...
本学习笔记将深入探讨JUC的基础知识,帮助你理解和掌握Java并发编程的核心概念。 在Java中,多线程是实现并发的主要方式。一个线程是程序执行的一个流,多线程则意味着一个程序内可以同时执行多个线程。Java提供了...
以上就是Java源码笔记可能涉及的主要内容,通过深入学习这些知识点,开发者可以更好地理解和运用Java进行网络编程,提高软件开发的效率和质量。同时,对源码的深入理解也有助于解决实际问题,提升编程技能。
3. **线程池**:Executor框架提供了线程池,有效管理线程生命周期,减少创建和销毁线程的开销。 以上是Java编程的一些核心知识点,涵盖了从基础语法到高级特性的各个方面。深入理解和掌握这些内容对于日常的Java...
通过阅读《Java JDK 1.60学习笔记.chm》这样的资源,开发者可以深入学习这些主题,进一步提升自己的Java编程能力。这份文档可能涵盖了所有这些知识点的详细讲解,以及实例代码和最佳实践,对于初学者和经验丰富的...
《Java并发编程实践》学习笔记 Java并发编程是Java开发者必须掌握的重要技能,尤其是在多核处理器和高并发应用中。本书《Java Concurrency In Practice》是Java并发编程的经典之作,由Brian Goetz、Tim Peierls、...
这份学习笔记将围绕JDK 6展开,涵盖其主要特性和基础Java知识。 1. **Java基础** - **语法**:Java是一种面向对象的编程语言,具有严格的类型检查和垃圾回收机制,保证了代码的稳定性和安全性。 - **类与对象**:...
Java还提供了高级的并发工具,如Executor框架、Semaphore、CountDownLatch等,用于更有效地管理线程和同步。 3. **IO流**:Java的IO流允许程序进行输入和输出操作,无论是处理文件、网络数据还是内存中的数据。IO流...
Java的Executor框架提供了线程池管理,通过`Executors`静态工厂方法创建线程池,如`newFixedThreadPool`创建固定大小的线程池,`newSingleThreadExecutor`创建只有一个工作线程的线程池。线程池能有效控制运行的线程...