实现最简单的分布式任务调度框架

开发 项目管理 分布式
前段时间,公司要改造现有的单节点调度为分布式任务调度,然后就研究了目前市面上主流的开源分布式任务调度框架,用起来就一个感觉:麻烦!特别是之前在一个类里写了好多个调度任务,改造起来更加麻烦。

 前段时间,公司要改造现有的单节点调度为分布式任务调度,然后就研究了目前市面上主流的开源分布式任务调度框架,用起来就一个感觉:麻烦!特别是之前在一个类里写了好多个调度任务,改造起来更加麻烦。我这人又比较懒,总感觉用了别人写好的工具还要改一大堆,心里就有点不舒服。

[[282425]]

于是我就想自己写一个框架,毕竟自己觉得分布式任务调度在所有分布式系统中是最简单的,因为一般公司任务调度本身不可能同时调度海量的任务,很大的并发,改造成分布式主要还是为了分散任务到多个节点,以便同一时间处理更多的任务。

后面有一天,我在公司前台取快递,看到这样一个现象:我们好几个同事(包括我)在前台那从头到尾看快递是不是自己的,是自己的就取走,不是就忽略,然后我就收到了启发。这个场景类比到分布式调度系统中,我们可以认为是快递公司或者快递员已经把每个快递按照我们名字电话分好了快递,我们只需要取走自己的就行了。

但是从另外一个角度看,也可以理解成我们每个人都是从头到尾看了所有快递,然后按照某种约定的规则,如果是自己的快递就拿走,不是自己的就忽略继续看下一个。如果把快递想象成任务,一堆人去拿一堆快递也可以很顺利的拿到各自的快递,那么一堆节点自己去取任务是不是也可以很好的处理各自的任务呢?

传统的分布式任务调度都有一个调度中心,这个调度中心也都要部署称多节点的集群,避免单点故障,然后还有一堆执行器,执行器负责执行调度中心分发的任务。按照上面的启发,我的思路是放弃中心式的调度中心直接由各个执行器节点去公共的地方按照约定的规则去取任务,然后执行。设计示意图如下

 

有人可能怀疑那任务db库不是有单点问题吗,我想反问下,难道其他的分布式任务调度框架没有这个问题吗?针对数据库单点我们可以单独类似业务库那样考虑高可用方案,这里不是这篇文章的讨论重点。很明显我们重点放在执行节点那里到底怎么保证高可用,单个任务不会被多个节点同时执行,单个节点执行到一半突然失联了,这个任务怎么办等复杂的问题。

后续我们使用未经修饰的代码的方式一一解决这个问题(未经修饰主要是没有优化结构流水账式的代码风格,主要是很多人包括我自己看别人源码时总是感觉晕头转向的,仿佛置身迷宫般,看起来特别费劲,可能是我自己境界未到吧)

既然省略了集中式的调度,那么既然叫任务调度很明显必须要有调度的过程,不然多个节点去抢一个任务怎么避免冲突呢?我这里解决方式是:首先先明确一个任务的几种状态:待执行,执行中,有异常,已完成。

每个节点起一个线程一直去查很快就要开始执行的待执行任务,然后遍历这些任务,使用乐观锁的方式先更新这个任务的版本号(版本号+1)和状态(变成执行中),如果更新成功就放入节点自己的延时队列中等待执行。

由于每个节点的线程都是去数据库查待执行的任务,很明显变成执行中的任务下次就不会被其他节点再查询到了,至于对于那些在本节点更新状态之前就查到的待执行任务也会经过乐观锁尝试后更新失败从而跳过这个任务,这样就可以避免一个任务同时被多个节点重复执行。关键代码如下:

  1. package com.rdpaas.task.scheduler; 
  2.  
  3. import com.rdpaas.task.common.*; 
  4. import com.rdpaas.task.config.EasyJobConfig; 
  5. import com.rdpaas.task.repository.NodeRepository; 
  6. import com.rdpaas.task.repository.TaskRepository; 
  7. import com.rdpaas.task.strategy.Strategy; 
  8.  
  9. import org.slf4j.Logger; 
  10. import org.slf4j.LoggerFactory; 
  11. import org.springframework.beans.factory.annotation.Autowired; 
  12. import org.springframework.stereotype.Component; 
  13.  
  14. import javax.annotation.PostConstruct; 
  15.  
  16. import java.util.Date
  17. import java.util.List; 
  18. import java.util.concurrent.*; 
  19.  
  20. /** 
  21.  * 任务调度器 
  22.  * @author rongdi 
  23.  * @date 2019-03-13 21:15 
  24.  */ 
  25. @Component 
  26. public class TaskExecutor { 
  27.  
  28.     private static final Logger logger = LoggerFactory.getLogger(TaskExecutor.class); 
  29.  
  30.     @Autowired 
  31.     private TaskRepository taskRepository; 
  32.  
  33.     @Autowired 
  34.     private NodeRepository nodeRepository; 
  35.  
  36.     @Autowired 
  37.     private EasyJobConfig config;/** 
  38.      * 创建任务到期延时队列 
  39.       */ 
  40.     private DelayQueue<DelayItem<Task>> taskQueue = new DelayQueue<>(); 
  41.  
  42.     /** 
  43.      * 可以明确知道最多只会运行2个线程,直接使用系统自带工具就可以了 
  44.      */ 
  45.     private ExecutorService bossPool = Executors.newFixedThreadPool(2); 
  46.  
  47.     /** 
  48.      * 声明工作线程池 
  49.      */ 
  50.     private ThreadPoolExecutor workerPool; 
  51.  
  52.  
  53.     @PostConstruct 
  54.     public void init() { 
  55. /** 
  56.          * 自定义线程池,初始线程数量corePoolSize,线程池等待队列大小queueSize,当初始线程都有任务,并且等待队列满后 
  57.          * 线程数量会自动扩充最大线程数maxSize,当新扩充的线程空闲60s后自动回收.自定义线程池是因为Executors那几个线程工具 
  58.          * 各有各的弊端,不适合生产使用 
  59.          */ 
  60.         workerPool = new ThreadPoolExecutor(config.getCorePoolSize(), config.getMaxPoolSize(), 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(config.getQueueSize())); 
  61.         /** 
  62.          * 执行待处理任务加载线程 
  63.          */ 
  64.         bossPool.execute(new Loader()); 
  65.         /** 
  66.          * 执行任务调度线程 
  67.          */ 
  68.         bossPool.execute(new Boss()); 
  69.  
  70.     } 
  71.  
  72.     class Loader implements Runnable { 
  73.  
  74.         @Override 
  75.         public void run() { 
  76.             for(;;) { 
  77.                 try {  
  78.             /** 
  79.                      * 查找还有指定时间(单位秒)开始的主任务列表 
  80.                      */ 
  81.                     List<Task> tasks = taskRepository.listPeddingTasks(config.getFetchDuration()); 
  82.                     if(tasks == null || tasks.isEmpty()) { 
  83.                         continue
  84.                     } 
  85.                     for(Task task:tasks) { 
  86.  
  87.                         task.setStatus(TaskStatus.DOING); 
  88.                         task.setNodeId(config.getNodeId()); 
  89.                         /** 
  90.                          * 使用乐观锁尝试更新状态,如果更新成功,其他节点就不会更新成功。如果在查询待执行任务列表 
  91.                          * 和当前这段时间有节点已经更新了这个任务,version必然和查出来时候的version不一样了,这里更新 
  92.                          * 必然会返回0了 
  93.                          */ 
  94.                         int n = taskRepository.updateWithVersion(task); 
  95.                         Date nextStartTime = task.getNextStartTime(); 
  96.                         if(n == 0 || nextStartTime == null) { 
  97.                             continue
  98.                         } 
  99.                         /** 
  100.                          * 封装成延时对象放入延时队列 
  101.                          */ 
  102.                         task = taskRepository.get(task.getId()); 
  103.                         DelayItem<Task> delayItem = new DelayItem<Task>(nextStartTime.getTime() - new Date().getTime(), task); 
  104.                         taskQueue.offer(delayItem); 
  105.  
  106.                     } 
  107.                     Thread.sleep(config.getFetchPeriod()); 
  108.                 } catch(Exception e) { 
  109.                     logger.error("fetch task list failed,cause by:{}", e); 
  110.                 } 
  111.             } 
  112.         } 
  113.  
  114.     } 
  115.  
  116.     class Boss implements Runnable { 
  117.         @Override 
  118.         public void run() { 
  119.             for (;;) { 
  120.                 try { 
  121.                      /** 
  122.                      * 时间到了就可以从延时队列拿出任务对象,然后交给worker线程池去执行 
  123.                      */ 
  124.                     DelayItem<Task> item = taskQueue.take(); 
  125.                     if(item != null && item.getItem() != null) { 
  126.                         Task task = item.getItem(); 
  127.                         workerPool.execute(new Worker(task)); 
  128.                     } 
  129.  
  130.                 } catch (Exception e) { 
  131.                     logger.error("fetch task failed,cause by:{}", e); 
  132.                 } 
  133.             } 
  134.         } 
  135.  
  136.     } 
  137.  
  138.     class Worker implements Runnable { 
  139.  
  140.         private Task task; 
  141.  
  142.         public Worker(Task task) { 
  143.             this.task = task; 
  144.         } 
  145.  
  146.         @Override 
  147.         public void run() { 
  148.             logger.info("Begin to execute task:{}",task.getId()); 
  149.             TaskDetail detail = null
  150.             try { 
  151.                 //开始任务 
  152.                 detail = taskRepository.start(task); 
  153.                 if(detail == nullreturn
  154.                 //执行任务 
  155.                 task.getInvokor().invoke(); 
  156.                 //完成任务 
  157.                 finish(task,detail); 
  158.                 logger.info("finished execute task:{}",task.getId()); 
  159.             } catch (Exception e) { 
  160.                 logger.error("execute task:{} error,cause by:{}",task.getId(), e); 
  161.                 try { 
  162.                     taskRepository.fail(task,detail,e.getCause().getMessage()); 
  163.                 } catch(Exception e1) { 
  164.                     logger.error("fail task:{} error,cause by:{}",task.getId(), e); 
  165.                 } 
  166.             } 
  167.         } 
  168.  
  169.     } 
  170.  
  171.     /** 
  172.      * 完成子任务,如果父任务失败了,子任务不会执行 
  173.      * @param task 
  174.      * @param detail 
  175.      * @throws Exception 
  176.      */ 
  177.     private void finish(Task task,TaskDetail detail) throws Exception { 
  178.  
  179.         //查看是否有子类任务 
  180.         List<Task> childTasks = taskRepository.getChilds(task.getId()); 
  181.         if(childTasks == null || childTasks.isEmpty()) { 
  182.             //当没有子任务时完成父任务 
  183.             taskRepository.finish(task,detail); 
  184.             return
  185.         } else { 
  186.             for (Task childTask : childTasks) { 
  187.                 //开始任务 
  188.                 TaskDetail childDetail = null
  189.                 try { 
  190.                     //将子任务状态改成执行中 
  191.                     childTask.setStatus(TaskStatus.DOING); 
  192.                     childTask.setNodeId(config.getNodeId()); 
  193.                     //开始子任务 
  194.                     childDetail = taskRepository.startChild(childTask,detail); 
  195.                     //使用乐观锁更新下状态,不然这里可能和恢复线程产生并发问题 
  196.                     int n = taskRepository.updateWithVersion(childTask); 
  197.                     if (n > 0) { 
  198.                         //再从数据库取一下,避免上面update修改后version不同步 
  199.                         childTask = taskRepository.get(childTask.getId()); 
  200.                         //执行子任务 
  201.                         childTask.getInvokor().invoke(); 
  202.                         //完成子任务 
  203.                         finish(childTask, childDetail); 
  204.                     } 
  205.                 } catch (Exception e) { 
  206.                     logger.error("execute child task error,cause by:{}", e); 
  207.                     try { 
  208.                         taskRepository.fail(childTask, childDetail, e.getCause().getMessage()); 
  209.                     } catch (Exception e1) { 
  210.                         logger.error("fail child task error,cause by:{}", e); 
  211.                     } 
  212.                 } 
  213.             } 
  214.             /** 
  215.              * 当有子任务时完成子任务后再完成父任务 
  216.              */ 
  217.             taskRepository.finish(task,detail); 
  218.  
  219.         } 
  220.  
  221.     } 
  222.  

如上所述,可以保证一个任务同一个时间只会被一个节点调度执行。这时候如果部署多个节点,正常应该可以很顺利的将任务库中的任务都执行到,就像一堆人去前台取快递一样,可以很顺利的拿走所有快递。毕竟对于每个快递不是自己的就是其他人的,自己的快递也不会是其他人的。

但是这里的调度和取快递有一点不一样,取快递的每个人都知道怎么去区分到底哪个快递是自己的。这里的调度完全没这个概念,完全是哪个节点运气好使用乐观锁更新了这个任务状态就是哪个节点的。总的来说区别就是需要一个约定的规则,快递是不是自己的,直接看快递上的名字和手机号码就知道了。任务到底该不该自己执行我们也可以出一个这种规则,明确哪些任务那些应该是哪些节点可以执行,从而避免无谓的锁竞争。

这里可以借鉴负载均衡的那些策略,目前我想实现如下规则:

  • id_hash : 按照任务自增id的对节点个数取余,余数值和当前节点的实时序号匹配,可以匹配就可以拿走执行,否则请自觉忽略掉这个任务
  • least_count:最少执行任务的节点优先去取任务
  • weight:按照节点权重去取任务
  • default:默认先到先得,没有其它规则

根据上面规则也可以说是任务的负载均衡策略可以知道除了默认规则,其余规则都需要知道全局的节点信息,比如节点执行次数,节点序号,节点权重等,所以我们需要给节点添加一个心跳,隔一个心跳周期上报一下自己的信息到数据库,心跳核心代码如下:

  1. /** 
  2.      * 创建节点心跳延时队列 
  3.       */ 
  4.     private DelayQueue<DelayItem<Node>> heartBeatQueue = new DelayQueue<>(); 
  5.    /** 
  6.      * 可以明确知道最多只会运行2个线程,直接使用系统自带工具 
  7.      */ 
  8.     private ExecutorService bossPool = Executors.newFixedThreadPool(2); 
  9.    
  10.  
  11.    @PostConstruct 
  12.     public void init() { 
  13.         /** 
  14.          * 如果恢复线程开关是开着,并且心跳开关也是开着 
  15.          */ 
  16.         if(config.isRecoverEnable() && config.isHeartBeatEnable()) { 
  17.             /** 
  18.              * 初始化一个节点到心跳队列,延时为0,用来注册节点 
  19.              */ 
  20.             heartBeatQueue.offer(new DelayItem<>(0,new Node(config.getNodeId()))); 
  21.             /** 
  22.              * 执行心跳线程 
  23.              */ 
  24.             bossPool.execute(new HeartBeat()); 
  25.             /** 
  26.              * 执行异常恢复线程 
  27.              */ 
  28.             bossPool.execute(new Recover()); 
  29.         } 
  30.     } 
  31.  
  32.    class HeartBeat implements Runnable { 
  33.         @Override 
  34.         public void run() { 
  35.             for(;;) { 
  36.                 try { 
  37.                     /** 
  38.                      * 时间到了就可以从延时队列拿出节点对象,然后更新时间和序号, 
  39.                      * 最后再新建一个超时时间为心跳时间的节点对象放入延时队列,形成循环的心跳 
  40.                      */ 
  41.                     DelayItem<Node> item = heartBeatQueue.take(); 
  42.                     if(item != null && item.getItem() != null) { 
  43.                         Node node = item.getItem(); 
  44.                         handHeartBeat(node); 
  45.                     } 
  46.                     heartBeatQueue.offer(new DelayItem<>(config.getHeartBeatSeconds() * 1000,new Node(config.getNodeId()))); 
  47.                 } catch (Exception e) { 
  48.                     logger.error("task heart beat error,cause by:{} ",e); 
  49.                 } 
  50.             } 
  51.         } 
  52.     } 
  53.  
  54.     /** 
  55.      * 处理节点心跳 
  56.      * @param node 
  57.      */ 
  58.     private void handHeartBeat(Node node) { 
  59.         if(node == null) { 
  60.             return
  61.         } 
  62.         /** 
  63.          * 先看看数据库是否存在这个节点 
  64.          * 如果不存在:先查找下一个序号,然后设置到node对象中,最后插入 
  65.          * 如果存在:直接根据nodeId更新当前节点的序号和时间 
  66.          */ 
  67.         Node currNode= nodeRepository.getByNodeId(node.getNodeId()); 
  68.         if(currNode == null) { 
  69.             node.setRownum(nodeRepository.getNextRownum()); 
  70.             nodeRepository.insert(node); 
  71.         } else  { 
  72.             nodeRepository.updateHeartBeat(node.getNodeId()); 
  73.         } 
  74.  
  75.     } 

数据库有了节点信息后,我们就可以实现各种花式的取任务的策略了,代码如下:

  1. /** 
  2.  * 抽象的策略接口 
  3.  * @author rongdi 
  4.  * @date 2019-03-16 12:36 
  5.  */ 
  6. public interface Strategy { 
  7.  
  8.     /** 
  9.      * 默认策略 
  10.      */ 
  11.     String DEFAULT = "default"
  12.  
  13.     /** 
  14.      * 按任务ID hash取余再和自己节点序号匹配 
  15.      */ 
  16.     String ID_HASH = "id_hash"
  17.  
  18.     /** 
  19.      * 最少执行次数 
  20.      */ 
  21.     String LEAST_COUNT = "least_count"
  22.  
  23.     /** 
  24.      * 按节点权重 
  25.      */ 
  26.     String WEIGHT = "weight"
  27.  
  28.  
  29.     public static Strategy choose(String key) { 
  30.         switch(key) { 
  31.             case ID_HASH: 
  32.                 return new IdHashStrategy(); 
  33.             case LEAST_COUNT: 
  34.                 return new LeastCountStrategy(); 
  35.             case WEIGHT: 
  36.                 return new WeightStrategy(); 
  37.             default
  38.                 return new DefaultStrategy(); 
  39.         } 
  40.     } 
  41.  
  42.     public boolean accept(List<Node> nodes,Task task,Long myNodeId); 
  43.  

~

  1. /** 
  2.  * 按照任务ID hash方式针对有效节点个数取余,然后余数+1后和各个节点的顺序号匹配, 
  3.  * 这种方式效果其实等同于轮询,因为任务id是自增的 
  4.  * @author rongdi 
  5.  * @date 2019-03-16 
  6.  */ 
  7. public class IdHashStrategy implements Strategy { 
  8.  
  9.     /** 
  10.      * 这里的nodes集合必然不会为空,外面调度那判断了,而且是按照nodeId的升序排列的 
  11.      */ 
  12.     @Override 
  13.     public boolean accept(List<Node> nodes, Task task, Long myNodeId) { 
  14.         int size = nodes.size(); 
  15.         long taskId = task.getId(); 
  16.         /** 
  17.          * 找到自己的节点 
  18.          */ 
  19.         Node myNode = nodes.stream().filter(node -> node.getNodeId() == myNodeId).findFirst().get(); 
  20.         return myNode == null ? false : (taskId % size) + 1 == myNode.getRownum(); 
  21.     } 
  22.  

~

  1. /** 
  2.  * 最少处理任务次数策略,也就是每次任务来了,看看自己是不是处理任务次数最少的,是就可以消费这个任务 
  3.  * @author rongdi 
  4.  * @date 2019-03-16 21:56 
  5.  */ 
  6. public class LeastCountStrategy implements Strategy { 
  7.  
  8.     @Override 
  9.     public boolean accept(List<Node> nodes, Task task, Long myNodeId) { 
  10.  
  11.         /** 
  12.          * 获取次数最少的那个节点,这里可以类比成先按counts升序排列然后取第一个元素 
  13.          * 然后是自己就返回true 
  14.          */ 
  15.         Optional<Node> min = nodes.stream().min((o1, o2) -> o1.getCounts().compareTo(o2.getCounts())); 
  16.  
  17.         return min.isPresent()? min.get().getNodeId() == myNodeId : false
  18.     } 
  19.  

~

  1. /** 
  2.  * 按权重的分配策略,方案如下,假如 
  3.  * 节点序号   1     ,2     ,3       ,4 
  4.  * 节点权重   2     ,3     ,3       ,2 
  5.  * 则取余后 0,1 | 2,3,4 | 5,6,7 | 8,9 
  6.  * 序号1可以消费按照权重的和取余后小于2的 
  7.  * 序号2可以消费按照权重的和取余后大于等于2小于2+3的 
  8.  * 序号3可以消费按照权重的和取余后大于等于2+3小于2+3+3的 
  9.  * 序号3可以消费按照权重的和取余后大于等于2+3+3小于2+3+3+2的 
  10.  * 总结:本节点可以消费的按照权重的和取余后大于等于前面节点的权重和小于包括自己的权重和的这个范围 
  11.  * 不知道有没有大神有更好的算法思路 
  12.  * @author rongdi 
  13.  * @date 2019-03-16 23:16 
  14.  */ 
  15. public class WeightStrategy implements Strategy { 
  16.  
  17.     @Override 
  18.     public boolean accept(List<Node> nodes, Task task, Long myNodeId) { 
  19.         Node myNode = nodes.stream().filter(node -> node.getNodeId() == myNodeId).findFirst().get(); 
  20.         if(myNode == null) { 
  21.             return false
  22.         } 
  23.         /** 
  24.          * 计算本节点序号前面的节点的权重和 
  25.          */ 
  26.         int preWeightSum = nodes.stream().filter(node -> node.getRownum() < myNode.getRownum()).collect(Collectors.summingInt(Node::getWeight)); 
  27.         /** 
  28.          * 计算全部权重的和 
  29.          */ 
  30.         int weightSum = nodes.stream().collect(Collectors.summingInt(Node::getWeight)); 
  31.         /** 
  32.          * 计算对权重和取余的余数 
  33.          */ 
  34.         int remainder = (int)(task.getId() % weightSum); 
  35.         return remainder >= preWeightSum && remainder < preWeightSum + myNode.getWeight(); 
  36.     } 
  37.  

然后我们再改造下调度类

  1. /** 
  2.      * 获取任务的策略 
  3.      */ 
  4.     private Strategy strategy; 
  5.  
  6.  
  7.     @PostConstruct 
  8.     public void init() { 
  9.         /** 
  10.          * 根据配置选择一个节点获取任务的策略 
  11.          */ 
  12.         strategy = Strategy.choose(config.getNodeStrategy()); 
  13.         /** 
  14.          * 自定义线程池,初始线程数量corePoolSize,线程池等待队列大小queueSize,当初始线程都有任务,并且等待队列满后 
  15.          * 线程数量会自动扩充最大线程数maxSize,当新扩充的线程空闲60s后自动回收.自定义线程池是因为Executors那几个线程工具 
  16.          * 各有各的弊端,不适合生产使用 
  17.          */ 
  18.         workerPool = new ThreadPoolExecutor(config.getCorePoolSize(), config.getMaxPoolSize(), 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(config.getQueueSize())); 
  19.         /** 
  20.          * 执行待处理任务加载线程 
  21.          */ 
  22.         bossPool.execute(new Loader()); 
  23.         /** 
  24.          * 执行任务调度线程 
  25.          */ 
  26.         bossPool.execute(new Boss()); 
  27.  
  28.     } 
  29.  
  30.     class Loader implements Runnable { 
  31.  
  32.         @Override 
  33.         public void run() { 
  34.             for(;;) { 
  35.                 try {  
  36.                     /** 
  37.                      * 先获取可用的节点列表 
  38.                      */ 
  39.                     List<Node> nodes = nodeRepository.getEnableNodes(config.getHeartBeatSeconds() * 2); 
  40.                     if(nodes == null || nodes.isEmpty()) { 
  41.                         continue
  42.                     } 
  43.                     /** 
  44.                      * 查找还有指定时间(单位秒)开始的主任务列表 
  45.                      */ 
  46.                     List<Task> tasks = taskRepository.listPeddingTasks(config.getFetchDuration()); 
  47.                     if(tasks == null || tasks.isEmpty()) { 
  48.                         continue
  49.                     } 
  50.                     for(Task task:tasks) { 
  51.  
  52.                         boolean accept = strategy.accept(nodes, task, config.getNodeId()); 
  53.                         /** 
  54.                          * 不该自己拿就不要抢 
  55.                          */ 
  56.                         if(!accept) { 
  57.                             continue
  58.                         } 
  59.                         task.setStatus(TaskStatus.DOING); 
  60.                         task.setNodeId(config.getNodeId()); 
  61.                         /** 
  62.                          * 使用乐观锁尝试更新状态,如果更新成功,其他节点就不会更新成功。如果在查询待执行任务列表 
  63.                          * 和当前这段时间有节点已经更新了这个任务,version必然和查出来时候的version不一样了,这里更新 
  64.                          * 必然会返回0了 
  65.                          */ 
  66.                         int n = taskRepository.updateWithVersion(task); 
  67.                         Date nextStartTime = task.getNextStartTime(); 
  68.                         if(n == 0 || nextStartTime == null) { 
  69.                             continue
  70.                         } 
  71.                         /** 
  72.                          * 封装成延时对象放入延时队列 
  73.                          */ 
  74.                         task = taskRepository.get(task.getId()); 
  75.                         DelayItem<Task> delayItem = new DelayItem<Task>(nextStartTime.getTime() - new Date().getTime(), task); 
  76.                         taskQueue.offer(delayItem); 
  77.  
  78.                     } 
  79.                     Thread.sleep(config.getFetchPeriod()); 
  80.                 } catch(Exception e) { 
  81.                     logger.error("fetch task list failed,cause by:{}", e); 
  82.                 } 
  83.             } 
  84.         } 
  85.  
  86.     } 

如上可以通过各种花式的负载策略来平衡各个节点获取的任务,同时也可以显著降低各个节点对同一个任务的竞争。

但是还有个问题,假如某个节点拿到了任务更新成了执行中,执行到一半,没执行完也没发生异常,突然这个节点由于各种原因挂了,那么这时候这个任务永远没有机会再执行了。这就是传说中的占着茅坑不拉屎。

解决这个问题可以用最终一致系统常见的方法,异常恢复线程。在这种场景下只需要检测一下指定心跳超时时间(比如默认3个心跳周期)下没有更新心跳时间的节点所属的未完成任务,将这些任务状态重新恢复成待执行,并且下次执行时间改成当前就可以了。核心代码如下:

  1. class Recover implements Runnable { 
  2.         @Override 
  3.         public void run() { 
  4.             for (;;) { 
  5.                 try { 
  6.                     /** 
  7.                      * 查找需要恢复的任务,这里界定需要恢复的任务是任务还没完成,并且所属执行节点超过3个 
  8.                      * 心跳周期没有更新心跳时间。由于这些任务由于当时执行节点没有来得及执行完就挂了,所以 
  9.                      * 只需要把状态再改回待执行,并且下次执行时间改成当前时间,让任务再次被调度一次 
  10.                      */ 
  11.                     List<Task> tasks = taskRepository.listRecoverTasks(config.getHeartBeatSeconds() * 3); 
  12.                     if(tasks == null || tasks.isEmpty()) { 
  13.                         return
  14.                     } 
  15.                    /** 
  16.                     * 先获取可用的节点列表 
  17.                     */ 
  18.                    List<Node> nodes = nodeRepository.getEnableNodes(config.getHeartBeatSeconds() * 2); 
  19.                    if(nodes == null || nodes.isEmpty()) { 
  20.                        return
  21.                    } 
  22.                    long maxNodeId = nodes.get(nodes.size() - 1).getNodeId(); 
  23.                     for (Task task : tasks) { 
  24.                         /** 
  25.                          * 每个节点有一个恢复线程,为了避免不必要的竞争,从可用节点找到一个最靠近任务所属节点的节点 
  26.                          */ 
  27.                         long currNodeId = chooseNodeId(nodes,maxNodeId,task.getNodeId()); 
  28.                         long myNodeId = config.getNodeId(); 
  29.                         /** 
  30.                          * 如果不该当前节点处理直接跳过 
  31.                          */ 
  32.                         if(currNodeId != myNodeId) { 
  33.                             continue
  34.                         } 
  35.                         /** 
  36.                          * 直接将任务状态改成待执行,并且节点改成当前节点 
  37.                          */ 
  38.                         task.setStatus(TaskStatus.PENDING); 
  39.                         task.setNextStartTime(new Date()); 
  40.                         task.setNodeId(config.getNodeId()); 
  41.                         taskRepository.updateWithVersion(task); 
  42.                     } 
  43.                     Thread.sleep(config.getRecoverSeconds() * 1000); 
  44.                 } catch (Exception e) { 
  45.                     logger.error("Get next task failed,cause by:{}", e); 
  46.                 } 
  47.             } 
  48.         } 
  49.  
  50.     } 
  51.   /** 
  52.      * 选择下一个节点 
  53.      * @param nodes 
  54.      * @param maxNodeId 
  55.      * @param nodeId 
  56.      * @return 
  57.      */ 
  58.     private long chooseNodeId(List<Node> nodes,long maxNodeId,long nodeId) { 
  59.         if(nodeId > maxNodeId) { 
  60.             return nodes.get(0).getNodeId(); 
  61.         } 
  62.         return nodes.stream().filter(node -> node.getNodeId() > nodeId).findFirst().get().getNodeId(); 
  63.     } 

如上为了避免每个节点的异常恢复线程对同一个任务做无谓的竞争,每个异常任务只能被任务所属节点ID的下一个正常节点去恢复。这样处理后就能确保就算出现了上面那种任务没执行完节点挂了的情况,一段时间后也可以自动恢复。

总的来说上面那些不考虑优化应该可以做为一个还不错的任务调度框架了。如果你们以为这样就完了,我只能说抱歉了,还有,哈哈!前面提到我是嫌弃其它任务调度用起来麻烦,特别是习惯用spring的注解写调度的,那些很可能一个类里写了n个带有@Scheduled注解的调度方法,这样改造起来更加麻烦,我是希望做到如下方式就可以直接整合到分布式任务调度里:

  1. /** 
  2.  * 测试调度功能 
  3.  * @author rongdi 
  4.  * @date 2019-03-17 16:54 
  5.  */ 
  6. @Component 
  7. public class SchedulerTest { 
  8.  
  9.     @Scheduled(cron = "0/10 * * * * ?"
  10.     public void test1() throws InterruptedException { 
  11.         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
  12.         Thread.sleep(2000); 
  13.         System.out.println("当前时间1:"+sdf.format(new Date())); 
  14.     } 
  15.  
  16.     @Scheduled(cron = "0/20 * * * * ?",parent = "test1"
  17.     public void test2() throws InterruptedException { 
  18.         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
  19.         Thread.sleep(2000); 
  20.         System.out.println("当前时间2:"+sdf.format(new Date())); 
  21.     } 
  22.  
  23.     @Scheduled(cron = "0/10 * * * * ?",parent = "test2"
  24.     public void test3() throws InterruptedException { 
  25.         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
  26.         Thread.sleep(2000); 
  27.         System.out.println("当前时间3:"+sdf.format(new Date())); 
  28.     } 
  29.  
  30.     @Scheduled(cron = "0/10 * * * * ?",parent = "test3"
  31.     public void test4() throws InterruptedException { 
  32.         SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 
  33.         Thread.sleep(2000); 
  34.         System.out.println("当前时间4:"+sdf.format(new Date())); 
  35.     } 
  36.  

为了达到上述目标,我们还需要在spring启动后加载自定义的注解(名称和spring的一样),代码如下

 

  1. /** 
  2.  * spring容器启动完后,加载自定义注解 
  3.  * @author rongdi 
  4.  * @date 2019-03-15 21:07 
  5.  */ 
  6. @Component 
  7. public class ContextRefreshedListener implements ApplicationListener<ContextRefreshedEvent> { 
  8.  
  9.     @Autowired 
  10.     private TaskExecutor taskExecutor; 
  11.  
  12.     /** 
  13.      * 用来保存方法名/任务名和任务插入后数据库的ID的映射,用来处理子任务新增用 
  14.      */ 
  15.     private Map<String,Long> taskIdMap = new HashMap<>(); 
  16.  
  17.     @Override 
  18.     public void onApplicationEvent(ContextRefreshedEvent event) { 
  19.         /** 
  20.          * 判断根容器为Spring容器,防止出现调用两次的情况(mvc加载也会触发一次) 
  21.           */ 
  22.         if(event.getApplicationContext().getParent()==null){ 
  23.             /** 
  24.              * 判断调度开关是否打开 
  25.              * 如果打开了:加载调度注解并将调度添加到调度管理中 
  26.              */ 
  27.             ApplicationContext context = event.getApplicationContext(); 
  28.             Map<String,Object> beans = context.getBeansWithAnnotation(org.springframework.scheduling.annotation.EnableScheduling.class); 
  29.             if(beans == null) { 
  30.                 return
  31.             } 
  32.             /** 
  33.              * 用来存放被调度注解修饰的方法名和Method的映射 
  34.              */ 
  35.             Map<String,Method> methodMap = new HashMap<>(); 
  36.             /** 
  37.              * 查找所有直接或者间接被Component注解修饰的类,因为不管Service,Controller等都包含了Component,也就是 
  38.              * 只要是被纳入了spring容器管理的类必然直接或者间接的被Component修饰 
  39.              */ 
  40.             Map<String,Object> allBeans = context.getBeansWithAnnotation(org.springframework.stereotype.Component.class); 
  41.             Set<Map.Entry<String,Object>> entrys = allBeans.entrySet(); 
  42.             /** 
  43.              * 遍历bean和里面的method找到被Scheduled注解修饰的方法,然后将任务放入任务调度里 
  44.              */ 
  45.             for(Map.Entry entry:entrys){ 
  46.                 Object obj = entry.getValue(); 
  47.                 Class clazz = obj.getClass(); 
  48.                 Method[] methods = clazz.getMethods(); 
  49.                 for(Method m:methods) { 
  50.                     if(m.isAnnotationPresent(Scheduled.class)) { 
  51.                         methodMap.put(clazz.getName() + Delimiters.DOT + m.getName(),m); 
  52.                     } 
  53.                 } 
  54.             } 
  55.             /** 
  56.              * 处理Sheduled注解 
  57.              */ 
  58.             handleSheduledAnn(methodMap); 
  59.             /** 
  60.              * 由于taskIdMap只是启动spring完成后使用一次,这里可以直接清空 
  61.              */ 
  62.             taskIdMap.clear(); 
  63.         } 
  64.     } 
  65.  
  66.     /** 
  67.      * 循环处理方法map中的所有Method 
  68.      * @param methodMap 
  69.      */ 
  70.     private void handleSheduledAnn(Map<String,Method> methodMap) { 
  71.         if(methodMap == null || methodMap.isEmpty()) { 
  72.             return
  73.         } 
  74.         Set<Map.Entry<String,Method>> entrys = methodMap.entrySet(); 
  75.         /** 
  76.          * 遍历bean和里面的method找到被Scheduled注解修饰的方法,然后将任务放入任务调度里 
  77.          */ 
  78.         for(Map.Entry<String,Method> entry:entrys){ 
  79.             Method m = entry.getValue(); 
  80.             try { 
  81.                 handleSheduledAnn(methodMap,m); 
  82.             } catch (Exception e) { 
  83.                 e.printStackTrace(); 
  84.                 continue
  85.             } 
  86.         } 
  87.     } 
  88.  
  89.     /** 
  90.      * 递归添加父子任务 
  91.      * @param methodMap 
  92.      * @param m 
  93.      * @throws Exception 
  94.      */ 
  95.     private void handleSheduledAnn(Map<String,Method> methodMap,Method m) throws Exception { 
  96.         Class<?> clazz = m.getDeclaringClass(); 
  97.         String name = m.getName(); 
  98.         Scheduled sAnn = m.getAnnotation(Scheduled.class); 
  99.         String cron = sAnn.cron(); 
  100.         String parent = sAnn.parent(); 
  101.         /** 
  102.          * 如果parent为空,说明该方法代表的任务是根任务,则添加到任务调度器中,并且保存在全局map中 
  103.          * 如果parent不为空,则表示是子任务,子任务需要知道父任务的id 
  104.          * 先根据parent里面代表的方法全名或者方法名(父任务方法和子任务方法在同一个类直接可以用方法名, 
  105.          * 不然要带上类的全名)从taskIdMap获取父任务ID 
  106.          * 如果找不到父任务ID,先根据父方法全名在methodMap找到父任务的method对象,调用本方法递归下去 
  107.          * 如果找到父任务ID,则添加子任务 
  108.          */ 
  109.         if(StringUtils.isEmpty(parent)) { 
  110.             if(!taskIdMap.containsKey(clazz.getName() + Delimiters.DOT + name)) { 
  111.                 Long taskId = taskExecutor.addTask(name, cron, new Invocation(clazz, name, new Class[]{}, new Object[]{})); 
  112.                 taskIdMap.put(clazz.getName() + Delimiters.DOT + name, taskId); 
  113.             } 
  114.         } else { 
  115.             String parentMethodName = parent.lastIndexOf(Delimiters.DOT) == -1 ? clazz.getName() + Delimiters.DOT + parent : parent; 
  116.             Long parentTaskId = taskIdMap.get(parentMethodName); 
  117.             if(parentTaskId == null) { 
  118.                 Method parentMethod = methodMap.get(parentMethodName); 
  119.                 handleSheduledAnn(methodMap,parentMethod); 
  120.                 /** 
  121.                  * 递归回来一定要更新一下这个父任务ID 
  122.                  */ 
  123.                 parentTaskId = taskIdMap.get(parentMethodName); 
  124.             } 
  125.             if(parentTaskId != null && !taskIdMap.containsKey(clazz.getName() + Delimiters.DOT + name)) { 
  126.                 Long taskId = taskExecutor.addChildTask(parentTaskId, name, cron, new Invocation(clazz, name, new Class[]{}, new Object[]{})); 
  127.                 taskIdMap.put(clazz.getName() + Delimiters.DOT + name, taskId); 
  128.             } 
  129.  
  130.         } 
  131.  
  132.  
  133.     } 

上述代码就完成了spring初始化完成后加载了自己的自定义任务调度的注解,并且也受spring的调度开关@EnableScheduling的控制,实现无缝整合到spring或者springboot中去,达到了我这种的懒人的要求。

好了其实写这个框架差不多就用了5天业余时间,估计会有一些隐藏的坑,不过明显的坑我自己都解决了,开源出来的目的既是为了抛砖引玉,也为了广大屌丝程序员提供一种新的思路,希望对大家有所帮助,同时也希望大家多帮忙找找bug,一起来完善这个东西,大神们请忽略。

责任编辑:武晓燕 来源: 博客园
相关推荐

2023-06-26 00:14:28

Openjob分布式任务

2023-05-08 16:38:46

任务调度分布式任务调度

2022-06-20 15:32:55

Stage模型分布式开发

2020-09-29 19:20:05

鸿蒙

2020-11-06 12:12:35

HarmonyOS

2019-07-19 15:51:11

框架选型分布式

2022-06-13 07:43:21

分布式Spring

2020-06-23 10:22:58

GitHub代码开发者

2024-05-23 10:19:57

2021-11-10 16:10:18

鸿蒙HarmonyOS应用

2022-03-28 07:51:25

分布式定时任务

2024-02-19 00:00:00

分布式定时任务框架

2022-08-09 08:40:37

框架分布式定时任务

2024-09-03 08:14:34

2024-09-23 04:00:00

java架构分布式系统

2021-11-29 08:48:00

K8S KubernetesAirflow

2020-03-31 08:05:23

分布式开发技术

2021-08-16 09:55:41

鸿蒙HarmonyOS应用

2023-04-19 16:51:54

分布式Primus开源

2010-06-03 19:46:44

Hadoop
点赞
收藏

51CTO技术栈公众号