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);
    }
}

TaskUtil多线程与定时任务的更多相关文章

  1. spring多线程与定时任务

    本篇主要描述一下spring的多线程的使用与定时任务的使用. 1.spring多线程任务的使用 spring通过任务执行器TaskExecutor来实现多线程与并发编程.通常使用ThreadPoolT ...

  2. Java多线程之定时任务(Timer)

    package org.study2.javabase.ThreadsDemo.schedule; import java.util.Date; import java.util.Timer; imp ...

  3. spring的@Scheduled定时任务,同一时间段的定时任务只会执行一个,其余的会被阻塞,@Scheduled注解定时任务并发执行解决办法,即多线程运行定时任务

    原文:https://blog.csdn.net/qq_35937303/article/details/88851064 现有两个定时任务 @Component("aa") pu ...

  4. Spring Boot 定时任务单线程和多线程

    Spring Boot 的定时任务: 第一种:把参数配置到.properties文件中: 代码: package com.accord.task; import java.text.SimpleDat ...

  5. Spring4.0编程式定时任务配置

    看过很多定时调度的配置,大多使用XML配置,觉得比较麻烦,也比较老套.这里介绍一种基于spring4.0注解编程式配置定时任务,简单清晰,使用方便.. 至于引入spring相关jar这里不多说,直接切 ...

  6. JAVA基础知识之多线程——线程组和未处理异常

    线程组 Java中的ThreadGroup类表示线程组,在创建新线程时,可以通过构造函数Thread(group...)来指定线程组. 线程组具有以下特征 如果没有显式指定线程组,则新线程属于默认线程 ...

  7. java多线程编程核心技术——第五章总结

    定时器Timer的使用 1.1方法schedule(TimerTask task, Date time)的测试 1.2方法schedule(TimerTask task, Date firstTime ...

  8. springBoot中使用定时任务

    简单示例 导入依赖 springBoot已经默认集成了定时任务的依赖,只需要引入基本的依赖就可以使用定时任务. <parent> <groupId>org.springfram ...

  9. Java编程的逻辑 (80) - 定时任务的那些坑

    ​本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http: ...

随机推荐

  1. HDU3874Necklace(树状数组+离线操作)

    此题的大意思说有一串珠子,每个珠子都有自己的欣赏值value,现在给你一串珠子每个的欣赏值,并给出一些询问,查询某个区间内部总欣赏值是多少,但是有一个约定就是如果这个区间内部有两个珠子的欣赏值是一样的 ...

  2. commondline 之一 常识

    用法: 执行类执行 jar 文件 java [-options] class [args...]java [-options] -jar jarfile [args...] 常识:javac Some ...

  3. 【转贴】Linux系统NGINX负载均衡404错误处理方法

    NGINX负载均衡404错误处理方法 使用NGINX 实现负载均衡,但一组服务器的数据不是实施同步,主服务器有了数据要过段时间才同步到其他服务器 upstream   image.stream.com ...

  4. android 简易定时器

    定时器 1.在android 应用开发当中,很多时候都要用到定时器,而要实现定时器更多的时候要用到两个类:Timer,和TimerTask 2.API对Timer的解释是:

  5. CT值及CT常用窗宽、窗位 [转]

    一.常用CT值 CT值的含义是:每个反应管内的荧光信号达到设定的域值时所经历的循环数.研究表明,每个模板的Ct值与该模板的起始拷贝数的 对数存在线性关系,起始拷贝数越多,Ct值越小.利用已知起始拷贝数 ...

  6. IE6/IE7下margin-bottom失效兼容解决办法及双倍边距问题

    (从已经死了一次又一次终于挂掉的百度空间人工抢救出来的,发表日期 2014-04-08) 一.IE6/IE7下margin-bottom失效兼容解决办法 1.用padding-bottom代替:2.在 ...

  7. 一条结合where、group、orderby的linq语法

    DataTable dt = (from x in dsResult.Tables[0].AsEnumerable() where DataTrans.CBoolean(x["IsCheck ...

  8. Flume-NG + HDFS + HIVE 日志收集分析

    国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...

  9. ArcSDE 10.2建立SDE服务

    从ArcGIS 10.1开始,arcgis官方推荐以直连方式连接SDE,因此在SDE安装时不再自动安装SDE服务,以下是手动安装SDE服务的方法 环境 服务端: oracle 11.2 64位,Arc ...

  10. 从零开始学android开发-项目重命名

    --修改项目名称 选中项目-[refactor]-[rename] --修改package名称 选中需要重命名的包-[refactor]-[rename] --修改gen下面的名称 打开Android ...