`
willsunforjava
  • 浏览: 167893 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

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代表一个有返回值得操作。

实例:并行计算求和

 

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();
	}
}
  • 大小: 6.6 KB
分享到:
评论

相关推荐

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

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

    Java+JDK+6学习笔记

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

    JAVA课程学习笔记.doc

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

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

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

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

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

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

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

    java学习笔记

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

    java学习笔记.7z

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

    java多线程学习笔记

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

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

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

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

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

    日常工作学习笔记(Java)

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

    java源码笔记

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

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

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

    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