package com.taoban.util;
/**
* 执行单次任务或定时任务工具类(用于减少new Thread()和new Timer()的使用)
*/
public class TaskUtil {
private static Log log = LogFactory.getLog(TaskUtil.class);
private static ExecutorService cachedExecutor = null;
private static ScheduledExecutorService scheduledExecutor = null;
private static Map<Runnable, Future<?>> keepRunningTasks = null;
private static Map<Future<?>, Callback> callbackdTasks = null;
static {
cachedExecutor = Executors.newCachedThreadPool(new TaskUtilThreadFactory("cached"));
scheduledExecutor = Executors.newScheduledThreadPool(5, new TaskUtilThreadFactory("scheduled"));
Runtime.getRuntime().addShutdownHook(new Thread() {//线程池自动退出
@Override
public void run() {
cachedExecutor.shutdown();
scheduledExecutor.shutdown();
log.info("TaskUtil executors shutdown.");
}
});
}
/**
* 立即执行任务
*/
public static Future<?> submit(Runnable task) {
return cachedExecutor.submit(task);
}
/**
* 自动保持任务持续运行,每分钟监视一次
*/
public static Future<?> submitKeepRunning(Runnable task){
Future<?> future = submit(task);
checkInitCachedTasks();
synchronized (keepRunningTasks) {
keepRunningTasks.put(task, future);
}
return future;
}
/**
* 延迟执行任务,例如延迟5秒:schedule(task,5,TimeUnit.SECONDS)
*/
public static void schedule(Runnable task, long delay, TimeUnit unit) {
scheduledExecutor.schedule(task, delay, unit);
}
/**
* 定时执行任务一次,比如下午两点:scheduleAt(task, DateUtils.setHours(new Date(), 13))
*/
public static void scheduleAt(Runnable task, Date time) {
long mills = time.getTime() - System.currentTimeMillis();
scheduledExecutor.schedule(task, mills>0 ? mills : 3, TimeUnit.MILLISECONDS);
}
/**
* 定时重复执行任务,比如延迟5秒,每10分钟执行一次:scheduleAtFixRate(task, 5, TimeUnit.MINUTES.toSeconds(10), TimeUnit.SECONDS)
*/
public static void scheduleAtFixtRate(Runnable task, long initialDelay, long delay, TimeUnit unit) {
scheduledExecutor.scheduleWithFixedDelay(task, initialDelay, delay, unit);
}
/**
* 定时重复执行任务,比如下午两点开始,每小时执行一次:scheduleAtFixRate(task, DateUtils.setHours(new Date(), 13), 1, TimeUnit.HOURS)
*/
public static void scheduleAtFixtRate(Runnable task, Date time, long delay, TimeUnit unit) {
long mills = time.getTime() - System.currentTimeMillis();
scheduledExecutor.scheduleWithFixedDelay(task, mills>0 ? mills : 3, unit.toMillis(delay), TimeUnit.MILLISECONDS);
}
/**
* 提交带返回值的任务,支持后续处理(调用者手动处理)
*/
public static <T> Future<T> submit(Callable<T> task) {
return cachedExecutor.submit(task);
}
/**
* 提交带返回值的任务,支持后续处理(自动调用Callback接口)
*/
public static <T> Future<T> submit(Callable<T> task, Callback callback) {
Future<T> future = submit(task);
checkInitCachedTasks();
if(callback != null) {
synchronized (callbackdTasks) {
callbackdTasks.put(future, callback);
}
}
return future;
}
/**
* 提交任务,等待返回值(阻塞调用者)
*/
public static <T> T wait(Callable<T> task) {
Future<T> future = cachedExecutor.submit(task);
try {
return future.get();
} catch (Exception e) {
log.warn(e);
return null;
}
}
private static void checkInitCachedTasks() {
if(keepRunningTasks != null) return;
keepRunningTasks = new HashMap<Runnable, Future<?>>();
callbackdTasks = new HashMap<Future<?>, Callback>();
scheduleAtFixtRate(new CachedTasksMonitor(), 1, 1, TimeUnit.MINUTES);
}
/**
* 监视需要保持运行的任务
*/
static class CachedTasksMonitor implements Runnable {
@Override
public void run() {
if(keepRunningTasks.size() > 0) {
synchronized (keepRunningTasks) {
Map<Runnable, Future<?>> tempTasks = null;
for(Runnable task : keepRunningTasks.keySet()) {
Future<?> future = keepRunningTasks.get(task);
if(future.isDone()) {
future = submit(task);//恢复运行异常结束任务
if(tempTasks == null) tempTasks = new HashMap<Runnable, Future<?>>();
tempTasks.put(task, future);
}
}
if(tempTasks != null && tempTasks.size() > 0) keepRunningTasks.putAll(tempTasks);
}
}
if(callbackdTasks.size() > 0) {
synchronized (callbackdTasks) {
List<Future<?>> callbackedFutures = null;
for(Future<?> future : callbackdTasks.keySet()) {
final Callback callback = callbackdTasks.get(future);
if(future.isDone()) {
try{
final Object result = future.get(5, TimeUnit.SECONDS);
submit(new Runnable() {
@Override
public void run() {//callback可能耗时所以作为独立运行任务,而本监视器需尽快完成工作
callback.handle(result);
}
});
if(callbackedFutures == null) callbackedFutures = new LinkedList<Future<?>>();
callbackedFutures.add(future);
}catch (Exception e) {
log.warn("TaskUtil callbackedTasks warn: ", e);
}
}
}
if(callbackedFutures != null && callbackedFutures.size() > 0) {
for(Future<?> future : callbackedFutures) {
callbackdTasks.remove(future);
}
}
}
}
}
}
/**
* 自定义线程名称Task-idx-name-idx2
*/
static class TaskUtilThreadFactory implements ThreadFactory {
private final static AtomicInteger taskutilThreadNumber = new AtomicInteger(1);
private final AtomicInteger threadNumber = new AtomicInteger(1);
private final String threadNamePrefix;
TaskUtilThreadFactory(String threadNamePrefix){
this.threadNamePrefix = threadNamePrefix;
}
@Override
public Thread newThread(Runnable r) {
Thread t = new Thread(r, String.format("TaskUtil-%d-%s-%d", taskutilThreadNumber.getAndIncrement(), this.threadNamePrefix, threadNumber.getAndIncrement()));
t.setDaemon(false);
t.setPriority(Thread.NORM_PRIORITY);
return t;
}
}
/**
* 等待结果回调接口
*/
public static interface Callback {
void handle(Object result);
}
}
- spring多线程与定时任务
本篇主要描述一下spring的多线程的使用与定时任务的使用. 1.spring多线程任务的使用 spring通过任务执行器TaskExecutor来实现多线程与并发编程.通常使用ThreadPoolT ...
- Java多线程之定时任务(Timer)
package org.study2.javabase.ThreadsDemo.schedule; import java.util.Date; import java.util.Timer; imp ...
- spring的@Scheduled定时任务,同一时间段的定时任务只会执行一个,其余的会被阻塞,@Scheduled注解定时任务并发执行解决办法,即多线程运行定时任务
原文:https://blog.csdn.net/qq_35937303/article/details/88851064 现有两个定时任务 @Component("aa") pu ...
- Spring Boot 定时任务单线程和多线程
Spring Boot 的定时任务: 第一种:把参数配置到.properties文件中: 代码: package com.accord.task; import java.text.SimpleDat ...
- Spring4.0编程式定时任务配置
看过很多定时调度的配置,大多使用XML配置,觉得比较麻烦,也比较老套.这里介绍一种基于spring4.0注解编程式配置定时任务,简单清晰,使用方便.. 至于引入spring相关jar这里不多说,直接切 ...
- JAVA基础知识之多线程——线程组和未处理异常
线程组 Java中的ThreadGroup类表示线程组,在创建新线程时,可以通过构造函数Thread(group...)来指定线程组. 线程组具有以下特征 如果没有显式指定线程组,则新线程属于默认线程 ...
- java多线程编程核心技术——第五章总结
定时器Timer的使用 1.1方法schedule(TimerTask task, Date time)的测试 1.2方法schedule(TimerTask task, Date firstTime ...
- springBoot中使用定时任务
简单示例 导入依赖 springBoot已经默认集成了定时任务的依赖,只需要引入基本的依赖就可以使用定时任务. <parent> <groupId>org.springfram ...
- Java编程的逻辑 (80) - 定时任务的那些坑
本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http: ...
随机推荐
- 【半原创】将js和css文件装入localStorage加速程序执行
首先感谢某某作者写的文章:http://www.jb51.net/article/12793.htm 直接上代码,注意文件名为env.js 原理如下: 一次批量加要加载的文件存入数组,采用Ajax方式 ...
- kindeditor 上传图片 显示绝对 路径
在前台页面上 效果图 另外附上 urlType 属性的 参数说明: 改变站内本地URL,可设置空.relative.absolute.domain. 空为不修改URL,relative为相对路径,ab ...
- android 工具类之SharePreference
/** * SharedPreferences的一个工具类,调用setParam就能保存String, Integer, Boolean, Float, Long类型的参数 * 同样调用getPara ...
- 高分辨率 2x图像: -webkit-min-device-pixel-ratio 的常见值对照
-webkit-min-device-pixel-ratio的常见值对照 原文地址:http://zhangyaochun.iteye.com/blog/1816582 前言: 本文来自于对 http ...
- 转载:mysql update更新带子查询的实现方式
出自:http://576017120.iteye.com/blog/1947154 mysql中更新时不能直接将更新的表作为查询的表,可以通过临时中间表的形式. 总结一下: 一:单表更新时 例如: ...
- position:sticky用法
用户的屏幕越来越大,而页面太宽的话会不宜阅读,所以绝大部分网站的主体宽度和之前相比没有太大的变化,于是浏览器中就有越来越多的空白区域,所以你可能注意到很多网站开始在滚动的时候让一部分内容保持可见,比如 ...
- 支付宝修改回调地址后 issign=false
原因: verifyReturn 拼接url的时候,php自动添加了url参数,而url是本不需要的,所以导致md5对比出错. 修改 alipay_notify.class.php 77行的函数 ...
- 【转】JAVA SSH 框架介绍
转自:http://www.admin10000.com/document/150.html SSH 为 struts+spring+hibernate 的一个集成框架,是目前较流行的一种JAVA W ...
- 从jQuery的缓存到事件监听
不知道大家有没有发现,用jQuery选择器"选择"之后的DOM上会添加jQuery*********属性. <DIV id=d1 jQuery1294122065250=&q ...
- 实战项目:通过当当API将订单抓取到SAP(一)
公司在当当上经营了一家店铺,通过当当提供的API,用C#写代码,通过NCO3.0调用SAP RFC将订单信息抓取到SAP. 如果你是新手,在当当网上有店铺,且你公司使用SAP系统,恭喜你,下面这些代码 ...