前言

前面介绍了Java并发包提供的三种线程池,它们用处各不相同,接下来介绍一些工具类,对这三种线程池的使用。

Executors

Executors是JDK1.5就开始存在是一个线程池工具类,它定义了用于Executor、ExecutorService、ScheduledExecutorService、ThreadFactory和Callable的工厂和工具方法。在开始之前,先了解一下它提供的一些内部类:

DelegatedExecutorService、DelegatedScheduledExecutorService、FinalizableDelegatedExecutorService

 //只暴露实现ExecutorService接口的方法的包装器类。Delegated 是代理,委托的意思
static class DelegatedExecutorService extends AbstractExecutorService {
private final ExecutorService e; //构造器传入一个ExecutorService实例
DelegatedExecutorService(ExecutorService executor) { e = executor; } .....
} //可自动终结的包装线程池,FinalizableDelegatedExecutorService的实例即使不手动调用shutdown方法关闭现称池,虚拟机也会帮你完成此任务
static class FinalizableDelegatedExecutorService extends DelegatedExecutorService { FinalizableDelegatedExecutorService(ExecutorService executor) {
super(executor);
} //finalize方法会在虚拟机gc清理对象时被调用
protected void finalize() {
super.shutdown();
}
} //只暴露实现ScheduledExecutorService的接口方法的一个包装器类。
static class DelegatedScheduledExecutorService extends DelegatedExecutorService implements ScheduledExecutorService {
private final ScheduledExecutorService e; //构造器传入一个ScheduledExecutorService实例
DelegatedScheduledExecutorService(ScheduledExecutorService executor) {
super(executor);
e = executor;
}
//.....
}

这三个类都是包装类,DelegatedExecutorService是对ExecutorService的一种包装,仅仅只给使用者暴露 ExecutorService的接口方法,屏蔽掉具体实现类的独有方法。DelegatedScheduledExecutorService是对ScheduledExecutorService的包装,仅仅只给使用者暴露 ScheduledExecutorService的接口方法,而FinalizableDelegatedExecutorService是在对ExecutorService的包装基础上,增加了自动线程池回收的功能,其finalize方法会在虚拟机gc清理对象时被调用,从而将用户忘记关闭的无用线程池关闭并回收。

PrivilegedCallableUsingCurrentClassLoader、PrivilegedCallable则是对Callable任务 的运行上下文和类加载的控制,RunnableAdapter则是用于将Runnable包装成Callable的包装类,DefaultThreadFactory是默认的线程工厂,创建的线程名称都具有:pool-池编号-thread-线程编号,这样的前缀。PrivilegedThreadFactory继承了DefaultThreadFactory,在默认的线程工厂上,扩展了捕获访问控制的上下文和类加载器。

工具方法

一、下面是一些创建线程池的工具方法:

public static ExecutorService newFixedThreadPool(int nThreads)

返回可以执行Runnable、Callable任务的一个固定线程池的ThreadPoolExecutor实例,其corePoolSize和maximumPoolSize都是指定的大小,keepAliveTime为0,任务队列是无界的LinkedBlockingQueue队列。首先根据 ThreadPoolExecutor的特性,corePoolSize和maximumPoolSize都相等时,意味着不会创建非核心线程,在keepAliveTime默认没有应用于核心线程时,其keepAliveTime无论是什么值都无意义,因此这里的keepAliveTime没有实际意义。然后由于是无界队列,maximumPoolSize参数其实也是无意义的,是所有来不及处理的任务都会无条件的丢进该无界队列中,直到系统资源耗尽,因此使用此线程池要注意任务的提交频率,不然会有内存耗尽,服务器宕机的风险。

public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory)

该方法只是比第一个方法newFixedThreadPool(int nThreads) 多传递了一个线程工厂参数,其他都一样。

public static ExecutorService newSingleThreadExecutor()

利用FinalizableDelegatedExecutorService返回包装过得可以执行Runnable、Callable任务的只有单个线程的ThreadPoolExecutor实例。其corePoolSize和maximumPoolSize都是1,任务队列是无界的LinkedBlockingQueue。首先,这是一个单线程按顺序执行任务的线程池,但如果在线程池关闭之前某个任务执行失败异常结束,那么如果需要执行后续任务,将可能会创建一个新的线程替代这个唯一的线程。其次该方法看似与执行newFixedThreadPool(1)的效果一样,但由于该方法放回的线程池经过FinalizableDelegatedExecutorService包装过,屏蔽了更改线程池配置的方法,因此该线程池无法被重新配置。最后经过FinalizableDelegatedExecutorService包装之后,该线程池具有了自动被JVM垃圾回收时终结回收的特性,即使用户使用完之后没有调用shutdown。

public static ExecutorService newSingleThreadExecutor(ThreadFactory threadFactory)

该方法就是比newSingleThreadExecutor()方法多传了一个线程工厂,其他都一样。

public static ExecutorService newCachedThreadPool()

返回可以立即执行Runnable、Callable任务的ThreadPoolExecutor线程池,其corePoolSize为0,maximumPoolSize为Integer.MAX_VALUE,但是任务队列是SynchronousQueue的0容量队列。因此,一旦任务被提交将立即被执行,如果线程不够将立即创建线程,该线程池理论上为了满足任务需要可以创建Integer.MAX_VALUE多个线程(当然该方法设置了keepAliveTime为60秒,在线程空闲下来之后所有线程都可能会被销毁),当任务的提交频率超过了任务的平均处理速率,将导致创建越来越多的线程以处理到达的任务,因此也有资源耗尽的潜在风险,必须要有效的控制任务的提交频率。

public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory)

指定了线程工厂的newCachedThreadPool(),其他都一样。

public static ScheduledExecutorService newSingleThreadScheduledExecutor()

返回一个可以执行Runnable、Callable类型的周期/延迟任务的的ScheduledThreadPoolExecutor线程池实例,其corePoolSize为1。由于其内部也是一个无界队列,因此maximumPoolSize、keepAliveTime是无效的。所以它也是一个单线程执行延迟/周期任务的线程池,所有来不及处理的任务都会无条件的丢进该无界队列中,直到系统资源耗尽,因此使用此线程池要注意任务的提交频率,不然会有内存耗尽,服务器宕机的风险。其返回经过了DelegatedScheduledExecutorService包装,不可再被转换成ScheduledThreadPoolExecutor实例。

public static ScheduledExecutorService newSingleThreadScheduledExecutor(ThreadFactory threadFactory)

指定了线程工厂的newSingleThreadScheduledExecutor(),其他都一样。

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize)

创建一个可以执行Runnable、Callable类型的周期/延迟任务的具有指定线程数量的ScheduledThreadPoolExecutor线程池实例,由于其内部是无界队列,所有来不及处理的任务都会无条件的丢进该无界队列中,直到系统资源耗尽,因此使用此线程池要注意任务的提交频率,不然会有内存耗尽,服务器宕机的风险。其返回结果没有经过包装,可以转换成ScheduledThreadPoolExecutor实例。

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize, ThreadFactory threadFactory)

指定了线程工厂的newScheduledThreadPool(int corePoolSize),其他都一样。

public static ExecutorService newWorkStealingPool(int parallelism)

返回一个可执行ForkJoinTask任务的具有给定并发度,工作线程按FIFO先进先出的异步工作模式的ForkJoinPool,由于ForkJoinPool是工作窃取机制实现的,所以该方法名有WorkStealing的字样。注意ForkJoinPool用于任务可递归分解执行的ForkJoinTask。由于该方法创建ForkJoinTask指定的asyncMode是ture,不能保证所提交任务的执行顺序,因此一般只适用于不需要合并递归执行结果(即不需要返回结果)的场景,一般用于事件消息传递的场景。

public static ExecutorService newWorkStealingPool()

无参的创建一个可执行ForkJoinTask任务的,工作线程按FIFO先进先出的异步工作模式的ForkJoinPool。该方法与newWorkStealingPool(int parallelism)唯一的区别就是,该方法的并行度是根据当前计算机的CPU核心数确定的,其他都一样,asyncMode也是true,一般来说这个方法比newWorkStealingPool(int parallelism)更好用。

对于以上提供的线程池的工具方法可以看出,Executors并没有返回基于有界队列的ThreadPoolExecutor线程池工具类,这大概是因为有界队列有任务被拒绝的潜在可能,这在大多数情况下用户都是不可接受的。在实际使用中,还是应该视情况而定,不要仅仅限于使用Executors提供的这几个工具方法返回的线程池,毕竟ThreadPoolExecutor线程池是一个可在多个参数上调节其工作机制的线程池。

二、下面是包装线程池的工具方法:

public static ExecutorService unconfigurableExecutorService(ExecutorService executor),包装指定的ExecutorService线程池为DelegatedExecutorService类型,即只暴露ExecutorService 的接口方法,其返回结果不能强转成原来的线程池实现类。按字面意思就是无法对该线程池的配置进行更改,因为已经将那些更改配置参数的setter方法都屏蔽了。

public static ScheduledExecutorService unconfigurableScheduledExecutorService(ScheduledExecutorService executor) ,包装指定的执行延迟/周期任务的线程池为DelegatedScheduledExecutorService类型,即只暴露ScheduledExecutorService的接口方法,其返回结果不能强转成原来的线程池实现类。按字面意思就是无法对该线程池的配置进行更改。

三、返回一个默认的线程工厂的工具方法:

public static ThreadFactory defaultThreadFactory(),返回默认的线程工厂实现DefaultThreadFactory,创建的线程名称都具有:pool-池编号-thread-线程编号,这样的前缀。创建出的线程都是非守护线程,因此使用该线程工厂的线程池在用完之后最好手动shutdown,避免JVM挂起。

public static ThreadFactory privilegedThreadFactory(),返回捕获了访问控制的上下文和类加载器的默认线程工厂DefaultThreadFactory的扩展类PrivilegedThreadFactory。创建的线程名称也都具有:pool-池编号-thread-线程编号,这样的前缀。

四,Callable相关的工具方法:

public static <T> Callable<T> callable(Runnable task, T result),将一个返回指定结果的Runnable 任务转换成Callable任务。

public static Callable<Object> callable(Runnable task) ,将一个返回结果为null的Runnable 任务转换成Callable任务。

public static Callable<Object> callable(final PrivilegedAction<?> action) ,返回一个在调用时执行指定的PrivilegedAction任务并返回其执行结果的Callable对象。

public static Callable<Object> callable(final PrivilegedExceptionAction<?> action) ,返回一个在调用时执行指定的PrivilegedExceptionAction任务并返回其执行结果的Callable对象。

public static <T> Callable<T> privilegedCallable(Callable<T> callable) ,  返回一个在调用它时可在当前的访问控制上下文中执行给定的Callable任务的Callable对象。

public static <T> Callable<T> privilegedCallableUsingCurrentClassLoader(Callable<T> callable), 返回一个在调用它时可在当前的访问控制上下文中,使用当前上下文类加载器作为上下文类加载器来执行给定的Callable任务的Callable对象。

CompletionService

这是一个接口,该接口是为了将任务的提交与获取任务的执行结果解耦而设计的,当然这里的任务一般是指多个。这通常应用于那种不关心执行顺序的多个耗时操作。例如异步I/O,或者类比一个页面的不同模块的异步加载。通常,CompletionService 依赖于一个单独的 Executor 来实际执行任务,在这种情况下,CompletionService 只管理一个内部完成队列。ExecutorCompletionService 类提供了此方法的一个实现,它虽然是一个需要创建实例的类,但其实也可以看作是一种工具类。

内存一致性效果:线程向 CompletionService 提交任务之前的操作 happen-before 该任务实际的执行操作,后者依次 happen-before 紧跟后面从完成队列中成功取走其返回结果的操作。

ExecutorCompletionService

首先看起其内部类QueueingFuture:

 public class ExecutorCompletionService<V> implements CompletionService<V> {
private final Executor executor;
private final AbstractExecutorService aes;
private final BlockingQueue<Future<V>> completionQueue; //已完成任务的Future阻塞队列 //将以完成的任务的Future排队的实现
private class QueueingFuture extends FutureTask<Void> { QueueingFuture(RunnableFuture<V> task) {
super(task, null);
this.task = task;
}
//实现回调接口,当任务被完成,将Future放进阻塞队列
protected void done() { completionQueue.add(task); }
private final Future<V> task;
} ......
}

QueueingFuture就是对FutureTask的扩展,利用了FutureTask任务在被执行完成之后会回调done()方法的特性,将已经完成的任务立即放到一个阻塞队列。

ExecutorCompletionService有两个构造方法,都需要传入一个Executor 线程池实现类,用来实际执行任务,另一个可选的参数就是内部使用的阻塞队列,阻塞队列的选取可以决定存取已完成任务的Future的顺序。

下面是它实现CompletionService 接口的方法:

 //提交Callable任务
public Future<V> submit(Callable<V> task) {
if (task == null) throw new NullPointerException();
RunnableFuture<V> f = newTaskFor(task);
executor.execute(new QueueingFuture(f));
return f;
} //提交Runnable任务,返回固定的结果
public Future<V> submit(Runnable task, V result) {
if (task == null) throw new NullPointerException();
RunnableFuture<V> f = newTaskFor(task, result);
executor.execute(new QueueingFuture(f));
return f;
} //获取一个已经完成的任务的Future,直到成功或者被中断。
public Future<V> take() throws InterruptedException {
return completionQueue.take();
} //获取并移除表示下一个已完成任务的 Future,如果不存在这样的任务,则返回 null。
public Future<V> poll() {
return completionQueue.poll();
} //超时版本的poll
public Future<V> poll(long timeout, TimeUnit unit)
throws InterruptedException {
return completionQueue.poll(timeout, unit);
}

从这些代码可见,都很简单,提交的任务被封装成QueueingFuture任务,从而可以使那些已经完成的任务被立即丢进准备好的阻塞队列,take/poll则是对阻塞队列中的已经完成的任务的Future的提取了。

以下是Java Doc中的两个示例,示例一:

 void solve(Executor e, Collection<Callable<Result>> solvers) throws InterruptedException, ExecutionException {
CompletionService ecs = new ExecutorCompletionService(e); //根据指定的线程池实例与任务集合创建ExecutorCompletionService实例
for (Callable s : solvers)
ecs.submit(s); //提交所有任务 int n = solvers.size();
for (int i = 0; i < n; ++i) {
Result r = ecs.take().get(); //提取已经完成的任务结果
if (r != null)
use(r); //对于非空返回结果执行相应的操作
}
}

示例一一次性提交一组任务,并且对每一个任务返回的非空结果执行给定的操作。

示例二:

 void solve(Executor e, Collection<Callable<Result>> solvers)
throws InterruptedException {
CompletionService ecs = new ExecutorCompletionService(e);
int n = solvers.size();
List<Future<Result>> futures = new ArrayList>(n);
Result result = null;
try {
for (Callable s : solvers)
futures.add(ecs.submit(s)); //提交所有任务,并记录Future for (int i = 0; i < n; ++i) {
try {
Result r = ecs.take().get();
if (r != null) { //发现一个任务完成就退出
result = r;
break;
}
} catch (ExecutionException ignore) {}
}
}
finally {
for (Future f : futures) //取消其余所有任务
f.cancel(true);
} if (result != null) //执行相应的操作
use(result);
}

示例二虽然也提交了一组任务,但只要有一个任务返回非空结果就取消其他所有任务,并那种该结果执行指定的操作。

ExecutorCompletionService利用了FutureTask任务不论是异常结束,还是正常结束还是被取消,都会回调其done方法的特点,扩展FutureTask并实现了该方法将不论以任何方式结束的任务的用于获取异步任务结果的Future放入一个阻塞队列中,因此可以通过读取队列的方法顺序获取那些任务的执行结果,由于任务可能是异常结束,因此在使用的时候,需要对Future.get()拿到的执行结果进行空值判断。

Java并发包线程池之Executors、ExecutorCompletionService工具类的更多相关文章

  1. java并发包&线程池原理分析&锁的深度化

          java并发包&线程池原理分析&锁的深度化 并发包 同步容器类 Vector与ArrayList区别 1.ArrayList是最常用的List实现类,内部是通过数组实现的, ...

  2. Java并发包——线程池

    Java并发包——线程池 摘要:本文主要学习了Java并发包中的线程池. 部分内容来自以下博客: https://www.cnblogs.com/dolphin0520/p/3932921.html ...

  3. Java并发包线程池之ForkJoinPool即ForkJoin框架(一)

    前言 这是Java并发包提供的最后一个线程池实现,也是最复杂的一个线程池.针对这一部分的代码太复杂,由于目前理解有限,只做简单介绍.通常大家说的Fork/Join框架其实就是指由ForkJoinPoo ...

  4. Java并发包线程池之ScheduledThreadPoolExecutor

    前言 它是一种可以安排在给定的延迟之后执行一次或周期性执行任务的ThreadPoolExecutor.因为它继承了ThreadPoolExecutor, 当然也具有处理普通Runnable.Calla ...

  5. Java并发包--线程池原理

    转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509954.html 线程池示例 在分析线程池之前,先看一个简单的线程池示例. 1 import jav ...

  6. Java并发包--线程池框架

    转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509903.html 线程池架构图 线程池的架构图如下: 1. Executor 它是"执行者 ...

  7. Java并发包线程池之ForkJoinPool即ForkJoin框架(二)

    前言 前面介绍了ForkJoinPool相关的两个类ForkJoinTask.ForkJoinWorkerThread,现在开始了解ForkJoinPool.ForkJoinPool也是实现了Exec ...

  8. Java并发包线程池之ThreadPoolExecutor

    参数详解 ExecutorService的最通用的线程池实现,ThreadPoolExecutor是一个支持通过配置一些参数达到满足不同使用场景的线程池实现,通常通过Executors的工厂方法进行配 ...

  9. java 多线程:线程池的使用Executors~ExecutorService; newCachedThreadPool;newFixedThreadPool(int threadNum);ScheduledExecutorService

    1,为什么要使用线程池:Executors 系统启动一个新线程的成本是比较高的,因为它涉及与操作系统交互.在这种情形下,使用线程池可以很好地提高性能,尤其是当程序中需要创建大量生存期很短暂的线程时,更 ...

随机推荐

  1. Django中使用geetest验证

    一.geetest的使用方法 首先需要在setting中配置好文件 GEE_TEST = { "gee_test_access_id": "37ca5631edd1e88 ...

  2. CSS之选择器相关

    一.选择器的作用 选择器就是用来选择标签的,要使用css对HTML页面中的元素实现一对一,一对多或者多对一的控制,这就需要用到CSS选择器. HTML页面中的元素就是通过CSS选择器进行控制的.每一条 ...

  3. 【转】Senior Data Structure · 浅谈线段树(Segment Tree)

    本文章转自洛谷 原作者: _皎月半洒花 一.简介线段树 ps: _此处以询问区间和为例.实际上线段树可以处理很多符合结合律的操作.(比如说加法,a[1]+a[2]+a[3]+a[4]=(a[1]+a[ ...

  4. python_面向对象——双下划线方法

    1.__str__和__repe__ class Person(object): def __init__(self,name,age): self.name = name self.age = ag ...

  5. JS转换HTML转义符 [转]

    最近有个需求,就是后台系统编辑文章内容存到后台,前端这边获取到是转义后的字符串,如果直接将转义后的内容写在页面上,html标签不会被解析.网上找到觉得不错的功能函数,这里记录一下 //去掉html标签 ...

  6. Selenium常用API的使用java语言之8-模拟鼠标操作

    通过前面例子了解到,可以使用click()来模拟鼠标的单击操作,现在的Web产品中提供了更丰富的鼠标交互方式, 例如鼠标右击.双击.悬停.甚至是鼠标拖动等功能.在WebDriver中,将这些关于鼠标操 ...

  7. Mac下mysql出现错误:ERROR 1055 (42000)

    问题原因: ONLY_FULL_GROUP_BY的意思是:对于GROUP BY聚合操作,如果在SELECT中的列,没有在GROUP BY中出现,那么这个SQL是不合法的,因为列不在GROUP BY从句 ...

  8. 如何让MySQL语句执行加速?

    一打开科技类论坛,最常看到的文章主题就是MySQL性能优化了,为什么要优化呢? 因为: 数据库出现瓶颈,系统的吞吐量出现访问速度慢 随着应用程序的运行,数据库的中的数据会越来越多,处理时间变长 数据读 ...

  9. IntelliJ IDEA——数据库集成工具(Database)的使用

    https://www.cnblogs.com/huiyi0521/p/10125537.html idea集成了一个数据库管理工具,可以可视化管理很多种类的数据库,意外的十分方便又好用.这里以ora ...

  10. sphinx和coreseek

    sphinx是国外的一款搜索软件. coreseek是在sphinx的基础上,增加了中文分词功能,换句话说,就是支持了中文. Coreseek发布了3.2.14版本和4.1版本,其中的3.2.14版本 ...