执行分段延时轮训任务

文章目录


背景

简单介绍下需求背景:

我对设备下发了一个指令,但是这个执行内容比较复杂,可能需要很久的时间才能完成,所以下发指令完成后,我需要轮询这个指令执行的状态,由于指令执行是一个过程,所以有多个状态,需要根据状态决定下次轮询间隔,直到给出结果(成功/失败都算结果)后,轮询结束;


一、什么是轮询

我去卖店买个可乐,店家说现在没有进货去了,很快就会到,我要了卖店的手机号,然后我就回家了。回家后,我隔一段时间(5分钟)就给店家打电话问,饮料来了吗?

我隔一段时间(10分钟)就给店家打电话问,饮料来了吗?

我隔一段时间(30分钟)就给店家打电话问,饮料来了吗?

我隔一段时间(60分钟)就给店家打电话问,饮料来了吗?

店家最终跟我说: 1.来了你来取吧 2.没有,今天饮料不能到了

这就是轮询

二、如何实现分段延时轮询

这里的分段指的是 根据状态,改变下次轮询的执行时间

这里大致说下思路:

  • 为了不影响主线程任务,这里采用新开启一个线程处理这个轮询
  • 开启这个任务线程后,每次执行一个延时任务,或者利用sleep 执行分段延时任务
  • 此场景中,每个任务必须等待上一个任务完成,判断状态,然后决定是否执行,以及执行的下个时间间隔
  • 当轮询过程中,执行状态已经明确给出结果(执行成功/执行失败),那么应该立即结束当前轮询任务

直接上代码

java 复制代码
import cn.hutool.core.date.DateUtil;
import cn.hutool.core.util.NumberUtil;
import cn.hutool.core.util.RandomUtil;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;

@Slf4j
public class Test {

    @SneakyThrows
    public static void main(String[] args) {

        log.info("开始执行主线程任务...");

        // 执行次数
        AtomicInteger atomicInteger = new AtomicInteger(0);
        // 是否需要取消任务
        AtomicBoolean shouldCancel = new AtomicBoolean(false);

        // 开始任务轮训
        ExecutorService executorService = Executors.newSingleThreadExecutor();

        try {
            for (int i = 0; i < 6; i++) {
                int decrement = atomicInteger.getAndIncrement();
                // 取消任务
                if (shouldCancel.get()) {
                    executorService.shutdownNow();
                    break;
                }
                Future<String> submit = executorService.submit(() -> {
                    log.info("子线程任务执行中...{}", DateUtil.now());
                    // 模拟返回任务状态
                    return RandomUtil.randomNumbers(2);
                });
                String res = submit.get();
                log.info("任务执行结果:{}", res);
                // 模拟任务执行成功/执行失败,其他状态为任务执行中间态,需要继续轮询
                if (res.equals("12") || res.equals("13")) {
                    executorService.shutdownNow();
                    shouldCancel.set(Boolean.TRUE);
                }
                // 等待时间 单位: 分 (这里为了加速,用秒)
                Thread.sleep(NumberUtil.pow(2, decrement).toBigInteger().longValue() * 1000);
    //            Thread.sleep(NumberUtil.pow(2, decrement).toBigInteger().longValue() * 1000 * 60);
                if (submit.isDone()) {
                    log.info("任务执行完毕...");
                }
            }
            executorService.shutdown();
            boolean over = !executorService.awaitTermination(70, TimeUnit.SECONDS);
            if (over) {
                executorService.shutdownNow(); // 如果等待超时,则取消所有未完成的任务
                System.out.println("结束");
                if (over)
                    System.err.println("线程池没有终止");
            }
        } finally {
            executorService.shutdown();
        }
    }

}

可以看到任务都是等待上一个任务执行,并且时间间隔也是按照我们的预想在一个个执行


总结

至此,就满足了这样一个分段延时的轮询任务的执行,而且这里是异步执行,主线程会立即返回,用户无感知的后台轮询,也节省了使用定时任务不停轮询造成的开销;
这里还有一个不用定时任务的原因:
除了上述说的造成开销外,就是定时任务有一定延迟,你如果设置定时任务的时间太短,那么服务器压力陡增,如果时间过长,那么会导致有一定的延迟,所以综合考量,都是这种方式更加合理

相关推荐
litGrey7 分钟前
Maven国内镜像(四种)
java·数据库·maven
学步_技术15 分钟前
Python编码系列—Python组合模式:构建灵活的对象组合
开发语言·python·组合模式
丶白泽33 分钟前
重修设计模式-结构型-桥接模式
java·设计模式·桥接模式
o独酌o39 分钟前
递归的‘浅’理解
java·开发语言
Book_熬夜!42 分钟前
Python基础(六)——PyEcharts数据可视化初级版
开发语言·python·信息可视化·echarts·数据可视化
无问8171 小时前
数据结构-排序(冒泡,选择,插入,希尔,快排,归并,堆排)
java·数据结构·排序算法
waterHBO1 小时前
R语言 基础笔记
开发语言·笔记·r语言
m0_631270401 小时前
高级c语言(五)
c语言·开发语言
customer081 小时前
【开源免费】基于SpringBoot+Vue.JS在线文档管理系统(JAVA毕业设计)
java·vue.js·spring boot·后端·开源
2401_858286111 小时前
53.【C语言】 字符函数和字符串函数(strcmp函数)
c语言·开发语言