`

ExecutorService的方法分析

阅读更多

ExecutorService接口继承了Executor接口其主要方法如下:

/**
	 * 优雅地关闭:先前提交的任务会继续执行完成,但不接受新的任务<br>
	 * 对于已关闭的ExecutorService没有影响
	 */
	void shutdown();

	/**
	 * 强行关闭:试图停止所有正在执行的任务,挂起等待执行的任务,返回等待执行任务的清单,而不是被中断任务的清单<br>
	 * 不能保证停止当前的任务会成功:典型的实现是通过调用Thread.interrupt()方法,所以不能回应该方法的任务将无法关闭
	 */
	List<Runnable> shutdownNow();

	/**
	 * 判断线程池是否已关闭<br>
	 * 在调用了shutdown()或shutdownNow()方法后,该方法返回true,此时线程池可能并未终止,调用isTerminated()
	 * 为false
	 * 
	 * @return
	 */
	boolean isShutdown();

	/**
	 * 判断线程池是否终止。只有调用shutdown()或shutdownNow()一定时间后,该方法才可能返回true
	 * 
	 * @return
	 */
	boolean isTerminated();

	/**
	 * 等待线程池终止。在所有任务执行完成,或者超时,或者当前线程被中断前,该方法一直阻塞
	 * 
	 * @param timeout
	 * @param unit
	 * @return
	 * @throws InterruptedException
	 */
	boolean awaitTermination(long timeout, TimeUnit unit)
			throws InterruptedException;

	/**
	 * 提交一个有返回结果的任务。如果成功完成,Future的get()方法会返回执行结果<br>
	 * Executors包含一系列方法,能够把其它普通的closure-like对象,转化成可以执行的Callable对象
	 * 
	 * @param <T>
	 * @param task
	 * @return
	 */
	<T> Future<T> submit(Callable<T> task);

	/**
	 * 提交一个有返回结果的任务。Runnable没有返回值,我们可以提供个预设值result,如果任务成功完成,将返回该result
	 * 
	 * @param <T>
	 * @param task
	 * @param result
	 * @return
	 */
	<T> Future<T> submit(Runnable task, T result);

	/**
	 * 提交一个有返回结果的任务。如果成功完成,Future.get()返回null。
	 * 
	 * @param task
	 * @return
	 */
	Future<?> submit(Runnable task);

	/**
	 * 执行指定的任务集,所有的任务完成后,返回一个保存了每个执行结果的Future集(可能是正常完成或者抛出异常)
	 * 
	 * @param <T>
	 * @param tasks
	 * @return
	 * @throws InterruptedException
	 *             被中断,没有完成的任务会被取消
	 * @throws NullPointerException
	 *             tasks为null,或者里面的元素为null
	 * @throws RejectedExecutionException
	 *             某个任务被拒绝执行
	 */
	<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks)
			throws InterruptedException;

	/**
	 * 执行指定的任务集,所有的任务完成后,返回一个保存了每个执行结果的Future集(可能是正常完成或者抛出异常)<br>
	 * 有时间限制:超时会返回,未完成的任务会取消
	 * 
	 * @param <T>
	 * @param tasks
	 * @param timeout
	 * @param unit
	 * @return
	 * @throws InterruptedException
	 */
	<T> List<Future<T>> invokeAll(Collection<? extends Callable<T>> tasks,
			long timeout, TimeUnit unit) throws InterruptedException;

	/**
	 * 执行指定的任务集,有一个任务执行成功后就返回该结果。正常返回或抛异常,没有完成的任务会被取消
	 * 
	 * @param <T>
	 * @param tasks
	 * @return
	 * @throws InterruptedException
	 *             被中断,没有完成的任务会被取消
	 * @throws ExecutionException
	 *             没有任务执行成功
	 */
	<T> T invokeAny(Collection<? extends Callable<T>> tasks)
			throws InterruptedException, ExecutionException;

	/**
	 * 执行指定的任务集,有一个任务执行成功后就返回该结果。正常返回或抛异常,没有完成的任务会被取消<br>
	 * 有时间限制
	 * 
	 * @param <T>
	 * @param tasks
	 * @return
	 * @throws InterruptedException
	 *             被中断,没有完成的任务会被取消
	 * @throws ExecutionException
	 *             没有任务执行成功
	 */
	<T> T invokeAny(Collection<? extends Callable<T>> tasks, long timeout,
			TimeUnit unit) throws InterruptedException, ExecutionException,
			TimeoutException;

 下面举例测试其中主要和常用的几个方法:

 首先,我们需要创建一个任务代码,这段任务代码主要是随机生成含有10个字符的字符串

  

/**
 * 随机生成10个字符的字符串
 * 
 * @author hegf
 * @date 2013-8-15 下午10:27:03
 */
public class Task1 implements Callable<String> {

	@Override
	public String call() throws Exception {
		String base = "abcdefghijklmnopqistuvwxyz0123456789";
		Random rand = new Random();
		StringBuffer buffer = new StringBuffer();
		for (int i = 0; i < 10; i++) {
			buffer.append(base.charAt(rand.nextInt(base.length())));
		}
		return buffer.toString();
	}

}

 然后,我们还需要一个长任务,这里我们默认是沉睡10秒

/**
 * 任务睡眠10秒的长任务
 * 
 * @author hegf
 * @date 2013-8-15 下午10:26:54
 */
public class LongTask implements Callable<String> {

	@Override
	public String call() throws Exception {
		TimeUnit.SECONDS.sleep(10);
		return null;
	}

}

 OK,所有前期准备完毕,下面我们就来分析一下ExecutorService接口中和生命周期有关的这些方法:

 

               ExecutorService service = Executors.newFixedThreadPool(4);
		service.submit(new Task1());
		service.submit(new Task1());
		service.submit(new LongTask());
		service.submit(new Task1());

		service.shutdown();
		while (!service.awaitTermination(1, TimeUnit.SECONDS)) {
			System.out.println("线程池未停止");
		}
		System.out.println("线程池已停止");

 这段代码中,我们在第三次提交了一个长任务,这个任务将执行10秒沉睡,紧跟着执行了一次shutdown()方法,假设:这时ExecutorService被立即关闭,下面调用service.awaitTermination(1, TimeUnit.SECONDS)方法时应该返回true,程序执行结果应该只会打印出:“线程池已关闭”。但是,真实的运行结果如下:

线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池未停止
线程池已停止

 这说明我们假设错误,service.awaitTermination(1, TimeUnit.SECONDS)每隔一秒监测一次ExecutorService的关闭情况,而长任务正好需要执行10秒,因此会在前9秒监测时ExecutorService为未关闭状态,而在第10秒时已经关闭,因此第10秒时输出:线程池已经关闭。这也验证了shutdown方法关闭ExecutorService的条件。

 

		ExecutorService service = Executors.newFixedThreadPool(3);
		service.submit(new LongTask());
		service.submit(new LongTask());
		service.submit(new LongTask());
		service.submit(new LongTask());
		service.submit(new LongTask());

		List<Runnable> runnables = service.shutdownNow();
		System.out.println(runnables.size());
		while (!service.awaitTermination(1, TimeUnit.SECONDS)) {
			System.out.println("线程池未停止");
		}
		System.out.println("线程池已停止");

  这段代码中,我们限制了线程池的长度是3,提交了5个长任务,这样将有两个任务在工作队列中等待,当我们执行shutdownNow方法时,ExecutorService被立刻关闭,所以在service.awaitTermination(1, TimeUnit.MILLISECONDS)方法校验时返回的是false,因此没有输出:线程池未关闭。而在调用shutdownNow方法时,我们接受到了一个List,这里包含的是在工作队列中等待执行的任务,由于线程池长度为3,且执行的都是长任务,所以当提交了三个任务后线程池已经满了,剩下的两次提交只能在工作队列中等待,因此我们看到runnables的大小为2,结果如下:

2
线程池已停止

 

		ExecutorService service = Executors.newFixedThreadPool(3);
		service.submit(new Task1());
		service.submit(new Task1());
		service.submit(new LongTask());

		service.shutdown();
		System.out.println(System.currentTimeMillis());
		while(!service.isTerminated()){
		}
		System.out.println(System.currentTimeMillis());

 这段代码我们执行了两个正常的任务和一个长任务,然后调用了shutdown方法,我们知道调用shutdown方法并不会立即关闭ExecutorService,这时我们记录一下监测循环执行前的时间,在没有关闭前我们一直进入一个空循环中,直到 ExecutorService关闭后退出循环,这里我们知道长任务执行时间为10秒,我们看一下上述程序运行结果:

1376579188593
1376579198593

 这10秒正好是长任务执行的时间,因此在 ExecutorService正常关闭后isTerminated方法返回true

分享到:
评论

相关推荐

    详细分析JAVA 线程池

    * ExecutorService newWorkStealingPool():该方法是前一个方法的简化版本。 6. ExecutorService接口 ExecutorService是Java中的一个接口,代表一个线程池,可以执行Runnable对象或Callable对象所代表的线程任务。...

    java 实例log分析

    - 并行处理:使用Java的并发工具类(如ExecutorService、Future)对日志进行多线程分析,提高处理速度。 - 缓存机制:对于频繁访问的数据,可以考虑引入缓存(如Guava Cache),减少不必要的计算和IO操作。 通过...

    数值分析.zip

    【数值分析.zip】这个压缩包文件包含了一个基于拉格朗日插值方法实现的图像缩放算法的Java项目。在IT领域,图像处理是计算机视觉和图形学中的一个重要分支,而拉格朗日插值是一种在多点之间进行数据插值的经典数学...

    JUC并发编程与源码分析视频课.zip

    4. **Executor框架**:介绍ExecutorService、ThreadPoolExecutor和ScheduledExecutorService等,理解任务调度和线程池的管理,学习如何合理配置线程池参数以优化系统性能。 5. **并发工具类**:包括CountDownLatch...

    Android应用异步编程模型性能分析.pdf

    本文针对这一问题,提出了基于异步跟踪的Android应用响应性能检测与分析方法,旨在解决由于异步编程导致的系统响应性能低和程序执行过程复杂性增加的问题。 首先,作者薛海龙、陈渝、雷蕾和王丹通过深入研究Android...

    java数值方法

    8. **错误分析和稳定性**:数值方法的一个关键方面是理解误差来源并确保算法的稳定性。这涉及到对算法的收敛性、舍入误差和截断误差的理解。 9. **复数运算**:尽管复数在物理和工程领域广泛应用,Java的`java.math...

    通过多线程任务处理大批量耗时业务并返回结果

    3. `submit()`方法:用于向`ExecutorService`提交任务,如果任务是`Callable`,它将返回一个`Future`,可以通过`Future.get()`来获取任务结果。 4. `CompletionService`:可能是`ExecutorCompletionService`,它...

    线程池核心组件源码剖析.docx

    3. **AbstractExecutorService抽象类**:这是ExecutorService的一个抽象实现,提供了`submit()`系列方法的实现。这些方法使用模板方法模式,内部调用了未实现的`execute()`方法,这个方法由具体的线程池实现类提供。...

    java程序员认证模拟题及详细分析

    3. 线程池:了解ExecutorService和ThreadPoolExecutor,理解线程池的工作原理。 六、反射 1. 获取类信息:使用Class类获取类的信息,如类名、字段、方法等。 2. 动态创建对象:通过newInstance()方法动态创建对象...

    Java 性能分析

    3. **使用并发库**:Java并发库提供了许多高效的并发工具,如ConcurrentHashMap、ExecutorService等,合理使用可提升多线程性能。 五、性能测试 1. **基准测试**:使用JMH (Java Microbenchmark Harness) 进行微...

    JDK自带线程池分析

    JDK自带线程池分析 JDK 自带线程池是 Java 语言中用于管理和执行线程的工具,旨在提高多线程编程的效率和灵活性。本文将详细介绍 JDK 自带线程池的组成、创建方法、优点和常见应用场景。 多线程技术 多线程技术是...

    基于Java swing带GUI界面的天气分析系统.rar

    这种分析方法可能涉及温度、湿度、风力、降雨量等多个气象参数,并将它们转换成易于理解和比较的百分制分数。此系统非常适合用作教学项目或课程设计,因为它为学生提供了一个实践Java GUI编程和数据处理的实例,同时...

    java多线程编程总结

    - 实现多线程最直接的方式之一就是继承`Thread`类并重写`run()`方法,该方法中包含了线程执行的具体逻辑。 **2. 示例代码** ```java public class MyThread extends Thread { @Override public void run() { ...

    Java Concurrent处理并发需求

    - 通过调用`executorService.submit()`方法,提交了三个Runnable对象给线程池。每个Runnable对象都封装了具体的任务逻辑,即调用`doSomething()`方法。 3. **任务执行**: - `doSomething()`方法模拟了一个耗时...

    Java 项目-基于网络爬虫技术的网络新闻分析.zip

    10. **文档编写**:编写项目报告,解释项目的设计思路、实现方法以及结果分析,这在毕业设计中是必不可少的。 这个项目提供了一个全面学习和实践Java网络爬虫及数据分析的好机会,不仅涵盖了编程技术,还涉及到了...

    Java rt.jar 源码分析

    5. **并发编程**:`java.util.concurrent`包包含了许多高级并发工具,如`ExecutorService`, `Future`, `Semaphore`, `CyclicBarrier`等。源码分析可以帮助我们理解这些工具的实现原理,从而更好地编写高效的并发程序...

    数据结构与算法分析_Java语言描述 第二版

    4. 并发编程:Java提供了丰富的并发工具,如ExecutorService、Semaphore、CyclicBarrier等,使得在多线程环境中实现算法变得更加安全和高效。 总结,《数据结构与算法分析——Java语言描述》第二版是一本深入浅出的...

    java 线程池管理类:Executors_.docx

    `java.util.concurrent.Executors` 继承自 `java.lang.Object`,作为一个工具类,它提供了一系列用于创建和管理线程池的方法,包括`ExecutorService`、`ScheduledExecutorService`、`ThreadFactory`和`Callable`等...

    多线程自定义规则 Nginx Access Log 分析器(比AWK快)

    3. **创建线程池**:使用ExecutorService创建线程池,设置合适的线程数量以充分利用CPU资源。 4. **分配任务**:为每个线程分配一个日志子集,并提供自定义的分析规则。 5. **处理结果**:线程执行完毕后,汇总并...

Global site tag (gtag.js) - Google Analytics