MoreExecutors是guava提供的工具类,是对jdk自带的Executors工具类的扩展,主要方法如下:

1、addDelayedShutDown()方法的两个重载:

public static void addDelayedShutDownHook(ExecutorService service, Duration terminationTimeout)

public static void addDelayedShutDownHook(ExecutorService service, long terminationTimeout, TimeUnit unit)

给线程池增加一个关闭钩子,在线程池中的线程是守护线程(daemon thread)时有用,用户线程(user thread)执行完后,jvm不会立即关闭,而是等待一定时间。正常情况下,只要用户线程一结束,jvm就会立即关闭,而不管守护线程任务是否执行完毕。从这里也可以看出,尽量不要把自定义线程搞成守护线程,不然是作死。

示例:

    public static void main(String[] args) {
ThreadFactory threadFactory = new ThreadFactoryBuilder().setDaemon(true).setNameFormat("async-pool-%d").build();
ExecutorService executorService = new ThreadPoolExecutor(10, 20, 0, TimeUnit.MINUTES, new LinkedBlockingQueue<>(3000), threadFactory);
executorService.submit(() -> {
try {
Thread.sleep(2000);
System.out.println(Thread.currentThread().getName() + "@666");
} catch (InterruptedException e) {
e.printStackTrace();
}
});
System.out.println(Thread.currentThread().getName() + "@888");
MoreExecutors.addDelayedShutdownHook(executorService, 3000, TimeUnit.MILLISECONDS);
}

上例中,如果没有调用MoreExecutors.addDelayedShutdownHook()方法的话,只会打印888,不会打印666。因为打印888的线程是用户线程,打印666的线程是守护线程(setDaemon(true)),用户线程一执行完,jvm就关闭了,所以不会有2000ms之后的666打印。假如创建ThreadFactory实例时,没有调用setDaemon(true),即创建的线程是非守护线程,那么会先打印main@888,2000ms后会打印async-pool-0@666,因为main线程和async-pool-0线程都是用户线程。调用MoreExecutors.addDelayedShutdownHook()方法后,jvm会在用户线程结束后等待一段时间再关闭,这段之间守护线程可以工作,到了时间,jvm关闭,守护线程也完蛋了,事情干多少是多少,没干完就没干完。

2、getExitingExecutorService()方法的三个重载:把一个ThreadPoolExecutor实例转成一个应用结束后自动退出的ExecutorService实例。

public static ExecutorService getExitingExecutorService(ThreadPoolExecutor executor)

public static ExecutorService getExitingExecutorService(ThreadPoolExecutor executor, Duration terminationTimeout)

public static ExecutorService getExitingExecutorService(ThreadPoolExecutor executor, long terminationTimeout, TimeUnit timeUnit)

示例:

    public static void main(String[] args) {
ThreadFactory threadFactory = new ThreadFactoryBuilder().setNameFormat("async-pool-%d").build();
ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(10, 20, 0, TimeUnit.MINUTES, new LinkedBlockingQueue<>(3000), threadFactory);
ExecutorService executorService = MoreExecutors.getExitingExecutorService(threadPoolExecutor);
executorService.submit(() -> {
try {
Thread.sleep(110000);
System.out.println(Thread.currentThread().getName() + "@666");
} catch (InterruptedException e) {
e.printStackTrace();
}
});
System.out.println(Thread.currentThread().getName() + "@888");
}

在用户线程执行完后,jvm会至多等待一定时间(默认是120s),以期线程池线程执行结束。这里不要求线程池中线程是守护线程,因为不管是不是守护线程,getExitingExecutorService()方法都会把这个线程池包装成守护线程的线程池,并且加个默认120s的关闭钩子。

3、getExitingScheduledExecutorService()方法的三个重载:道理同getExitingExecutorService()方法差不多

public static ScheduledExecutorService getExitingScheduledExecutorService(ScheduledThreadPoolExecutor executor)

public static ScheduledExecutorService getExitingScheduledExecutorService(ScheduledThreadPoolExecutor executor, Duration terminationTimeout)

public static ScheduledExecutorService getExitingScheduledExecutorService(ScheduledThreadPoolExecutor executor, long terminationTimeout, TimeUnit timeUnit)

4、listeningDecorator()方法的两个重载:

public static ListeningExecutorService listeningDecorator(ExecutorService delegate):把一个ExecutorService实例转成一个ListeningExecutorService实例

示例:

    public static void main(String[] args) {
ThreadFactory threadFactory = new ThreadFactoryBuilder().setNameFormat("async-pool-%d").build();
ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(10, 20, 0, TimeUnit.MINUTES, new LinkedBlockingQueue<>(3000), threadFactory);
ListeningExecutorService listeningExecutorService = MoreExecutors.listeningDecorator(threadPoolExecutor);
ListenableFuture future = listeningExecutorService.submit(() -> {
try {
Thread.sleep(4000);
System.out.println(Thread.currentThread().getName() + "@666");
} catch (InterruptedException e) {
e.printStackTrace();
}
}, 1);
Futures.addCallback(future, new FutureCallback() { @Override
public void onSuccess(Object result) {
System.out.println(Thread.currentThread().getName() + "@" + result);
} @Override
public void onFailure(Throwable t) {
System.out.println(Thread.currentThread().getName() + "@" + t.getMessage());
}
}, threadPoolExecutor);
System.out.println(Thread.currentThread().getName() + "@888");
}

ListeningExecutorService实例的submit()方法返回值是一个ListenableFuture实例,利用Futures工具类可以给这个实例添加callback回调。

public static ListeningScheduledExecutorService listeningDecorator(ScheduledExecutorService delegate):把一个ScheduledExecutorService实例转成一个ListeningScheduledExecutorService实例

5、directExector()方法:direct是直接的意思。

public static Executor directExecutor():返回一个Executor实例,具体是DirectExecutor类型,DirectExecutor是一个实现了Executor接口的枚举类。调用execute(Runnable command)方法时,在当前线程执行任务,而不会另起一个线程。这个方法没卵用,在当前线程执行的话,直接写代码就好了,干嘛还整这个。

6、newDirectExecutorService()方法

public static ListeningExecutorService newDirectExecutorService():返回一个ListeningExecutorService实例,具体是DirectExecutorService类型,DirectExecutorService是MoreExecutors的内部类,继承了AbstractListeningExecutorService。和DirectExecutor实例类似,调用DirectExecutorService实例的submit()方法时,会在当前线程执行任务,而不会另起一个线程。同样,没卵用。这里还要吐槽一下DirectExecutorService类名的不规范,妈的,明明是ListeningExecutorService实现类,就不能叫DirectListeningExecutorService吗。

7、newSequentialExecutor()方法:

public static Executor newSequentialExecutor(Executor delegate):把一个Executor实例包装成一个顺序执行的Executor实例,具体是SequentialExecutor类型。线程池按照任务添加顺序执行任务。

示例:

    public static void main(String[] args) {
ThreadFactory threadFactory = new ThreadFactoryBuilder().setNameFormat("async-pool-%d").build();
ThreadPoolExecutor threadPoolExecutor =
new ThreadPoolExecutor(10, 20, 0, TimeUnit.MINUTES, new LinkedBlockingQueue<>(3000), threadFactory);
Executor executor = MoreExecutors.newSequentialExecutor(threadPoolExecutor);
for (int i = 0; i < 10; i++) {
int d = i;
executor.execute(() -> {
try {
Thread.sleep(d * 1000);
System.out.println(
Thread.currentThread().getName() + "@" + d + ", now= " + System.currentTimeMillis());
} catch (InterruptedException e) {
e.printStackTrace();
}
});
}
}

线程池按照任务添加顺序执行任务。上一个任务执行不完,下一个任务就不会开始。活活把一个线程池包装成了一个单线程的线程池,鸡肋。

8、platformThreadFactory()方法:

MoreExecutors.platformThreadFactory():返回默认的ThreadFactory实例。内部调用Executors.defaultThreadFactory(),返回DefaultThreadFactory实例。DefaultThreadFactory是Executors的内部类,实现了ThreadFactory接口。DefaultThreadFactory对于的线程名形如pool-0-thread-0,pool-0-thread-1,我们自定义线程池时线程的名字绝对不可能这么没有意义,所以platformThreadFactory()这个方法也是鸡肋。

MoreExecutors工具类使用的更多相关文章

  1. Futures工具类使用

    Futures是guava提供的工具类,全类名是com.google.common.util.concurrent.Futures.配合MoreExecutors使用,效果极佳. 主要方法如下: 1. ...

  2. Java基础Map接口+Collections工具类

    1.Map中我们主要讲两个接口 HashMap  与   LinkedHashMap (1)其中LinkedHashMap是有序的  怎么存怎么取出来 我们讲一下Map的增删改查功能: /* * Ma ...

  3. Android—关于自定义对话框的工具类

    开发中有很多地方会用到自定义对话框,为了避免不必要的城府代码,在此总结出一个工具类. 弹出对话框的地方很多,但是都大同小异,不同无非就是提示内容或者图片不同,下面这个类是将提示内容和图片放到了自定义函 ...

  4. [转]Java常用工具类集合

    转自:http://blog.csdn.net/justdb/article/details/8653166 数据库连接工具类——仅仅获得连接对象 ConnDB.java package com.ut ...

  5. js常用工具类.

    一些js的工具类 复制代码 /** * Created by sevennight on 15-1-31. * js常用工具类 */ /** * 方法作用:[格式化时间] * 使用方法 * 示例: * ...

  6. Guava库介绍之实用工具类

    作者:Jack47 转载请保留作者和原文出处 欢迎关注我的微信公众账号程序员杰克,两边的文章会同步,也可以添加我的RSS订阅源. 本文是我写的Google开源的Java编程库Guava系列之一,主要介 ...

  7. Java程序员的日常—— Arrays工具类的使用

    这个类在日常的开发中,还是非常常用的.今天就总结一下Arrays工具类的常用方法.最常用的就是asList,sort,toStream,equals,copyOf了.另外可以深入学习下Arrays的排 ...

  8. .net使用正则表达式校验、匹配字符工具类

    开发程序离不开数据的校验,这里整理了一些数据的校验.匹配的方法: /// <summary> /// 字符(串)验证.匹配工具类 /// </summary> public c ...

  9. WebUtils-网络请求工具类

    网络请求工具类,大幅代码借鉴aplipay. using System; using System.Collections.Generic; using System.IO; using System ...

随机推荐

  1. 20191209 Linux就该这么学(4)

    4. Vim编辑器与Shell命令脚本 Vim 编辑器中设置了三种模式-命令模式.末行模式和编辑模式. 命令模式:控制光标移动,可对文本进行复制.粘贴.删除和查找等工作. 输入模式:正常的文本录入. ...

  2. 写 JSP 的痛点,真的非常痛!

    一.前戏 前后端分离已成为互联网项目开发的业界标准使用方式,通过nginx+tomcat的方式(也可以中间加一个nodejs)有效的进行解耦,并且前后端分离会为以后的大型分布式架构.弹性计算架构.微服 ...

  3. 【7.9校内test】T1挖地雷

    (土气的名字万里挑一丫丫) 然后这个题是个递推,lz的考场想法: 3个的最好确定,先把3个的确定下来,然后从这个点往前推,从这个点往后推这么算吧qwq 然后码长:也是很nice,最关键的是,我都写的这 ...

  4. 【7.9校内test】T2 极值问题

    这个题真的,毫无思路的说,但是我们会打表啊: lz的打表之路: 当然是手写一个暴力啦(*^▽^*)! 然后滚去配置lemon测试一下暴力可以得多少分qwq: 是的40分呢! 然后其实看上面也能看出来一 ...

  5. php中文网--JavaScript

    PHP中文网:http://www.php.cn/course/18.html 常用的两个客户端输出方法 document.write("你好呀js"); 描述:在网页的<b ...

  6. 洛谷 - P3225 - 矿场搭建 - 双连通分量

    https://www.luogu.org/problem/P3225 这个东西有点绕. 最平凡的情况,整个原图只有一个点,那么它坍塌了之后就没有点了,不需要进行任何逃生.否则,当一个点坍塌之后,每个 ...

  7. go相关资料

    1.go的调度2.go struct能不能比较 因为是强类型语言,所以不同类型的结构不能作比较,但是同一类型的实例值是可以比较的,实例不可以比较,因为是指针类型 3.go defer(for defe ...

  8. N4复习考试总结

    一つ(ひとつ) 半分(はんぶん) 煙草(たばこ)を吸う(すう) 玄関(げんかん) ナイフ(刀)     財布(さいふ) 浅い(あさい) 薄い(うすい) 牛乳(ぎゅうにゅう) 皿(さら) 七日(なのか) ...

  9. uni-app导航栏配置

    uni-app写app的内容会与沉浸栏重合在一起,写好好多,都是有点问题的,这次终于找到解决的方法了,与大家分享一下 最简单的解决方式就是配置mainfest.json来关闭沉浸式.即通过打开应用的m ...

  10. Linux文件读写笔记

    读文件: #include <stdio.h> #include <stdlib.h> #include <unistd.h> //linux下面的头文件 #inc ...