##目录
摘要: 实际项目中需要用到JOB的场景非常多,通常需要JOB的业务都需要如下基本功能: 1)集群环境单实例调度; 2)JOB可视化启停; 3)JOB频率可视化配置; 单应用使用JOB开发适配的需求的调度功能过滤繁琐,开销太大,考虑提供统一的调度系统对外提供JOB的调度服务。
# Marble简介
Marble是一款基于Quartz的计划任务调度系统,提供基本的任务定时调度功能。Marble包含两部分:
- 1)Marble OFFLINE: 一套可视化的JOB配置系统。包含JOB的启停、频率配置、调度日志查看等功能;
- 2)Marble AGENT:以Jar包形式整合进接入Marble的第三方应用,透明化Job的调度过程;详情见<< Marble使用>>章节
OFFLINE>> 为了可视化管理JOB,提供一个前端基于Bootstrap, 后端采用SpringMVC + MyBatis + MySql结构的控制台。提供基本的JOB的增删改查,启动/暂停等操作。
目前版本Marble提供的功能如下:
- 基本JOB管理(增删改查、暂停/启动);
- JOB执行参数传递的支持;
- 同步JOB和异步JOB支持;
- CRON表达式生成控件;
- 多种Misfire策略配置;
- 集群环境部署(不重复执行);
- 集群的软负载均衡;
- Spring整合;
服务端=>
- 引入相关Jar包(Netty等)
- 定义Job的具体实现类(Spring中的Bean)
- Spring中配置Job的实现类
OFFLINE端=>
- 配置对应应用和Job信息(配合权限管理)
####接入Marble - 服务端
- 引入相关Jar包(以Maven为例)
注:依赖marble-agent包(自己通过源代码的marble-server模块生成)和Thrift包
<!-- Marble相关 -->
<dependency>
<groupId>com.github.jxdong</groupId>
<artifactId>marble-agent</artifactId>
<version>1.0.1</version>
</dependency>
<!-- thrift相关(2.0.0之前版本需要) -->
<dependency>
<groupId>org.apache.thrift</groupId>
<artifactId>libthrift</artifactId>
<version>0.9.3</version>
</dependency>
<!-- Netty相关 -->
<dependency>
<groupId>io.netty</groupId>
<artifactId>netty-all</artifactId>
<version>4.0.4.Final</version>
</dependency>
2.定义Job具体实现类
定义java类继承父类MarbleJob,覆盖方法execute (当Job被调用时会执行execute下的内容)。
//异步JOB
@Component
public class TestMarbleJob1 extends MarbleJob {
private Logger logger = LoggerFactory.getLogger(TestMarbleJob1.class);
@Override
public void execute(String param) {
logger.info(" 异步 JOB1 执行正常: {}", param);
}
}
//同步JOB
@Component
public class TestSyncMarbleJob1 extends MarbleJob {
private Logger logger = LoggerFactory.getLogger(TestSyncMarbleJob1.class);
@Override
public Result executeSync(String param) {
logger.info(" 同步 JOB1(3S) 执行正常: {}", param);
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return Result.SUCCESS();
}
}
- Spring中添加Job配置
marble框架中已经自定义了Spring标签<marble ..>方便直接配置。 示例配置如下:
<marble:scheduler id="Scheduler001" port="9091" appCode="8890">
<marble:job name="Job1" description="jobDescription1" ref="testMarbleJob1"/>
<marble:job name="Job2" description="jobDescription2" ref="testSyncMarbleJob1"/>
</marble:scheduler>
配置解释:
配置了一个名称为Scheduler001的计划任务(job组),在9091端口上暴露服务。
Scheduler001下包含了两个Job:job1和job2.
Job1的实现类是 Spring bean - testMarbleJob1
Job2的实现类是Spring Bean – testSyncMarbleJob1
Marble标签详解:
<marble:scheduler/>: 计划任务配置。一个应用可以配置多个计划任务,每个计划任务暴露在一个IP的端口下,一个计划任务中可以包含多个Job。
属性 | 是否必填 | 描述 |
---|---|---|
id | 必填 | ID属性,英文+数字的组合来定义该计划任务的唯一标识。 |
host | 非必填 | 所在服务器的IP地址。指定后Marble会尝试将服务暴露在该IP下。不是必填项。由于集群,建议该字段不填,Marble默认会自动获取; |
port | 必填 | 计划任务暴露的端口号。指明该计划任务暴露在机器的哪个端口下,Marble随Spring启动后会尝试打开本机的该端口并暴露服务。 |
appCode | (必填) | 所在应用的APPID。为了在同一台机器上区别不同的应用,以免调用时混淆。 |
<marble:job/>: 计划任务下的job的配置。指明具体任务的别名以及实现类。
属性 | 是否必填 | 描述 |
---|---|---|
name | 必填 | 给当前Job起的别名,在所属的scheduler下必须唯一。 |
description | 非必填 | ob的描述信息,仅用来便于开发者自己识别,非必填。 |
ref | 必填 | Job的具体实现类,必须指向存在的一个Spring Bean。值为bean的name。且该bean必须继承自MarbleJob并覆盖execute方法。Job被调用时会执行execute方法,因此执行逻辑要写在execute中。 |
1、添加相关应用
- 应用Code:唯一的APPID
- 应用name:应用name必须与server端中marble:scheduler/标签的appCode属性值一致
- 应用描述
- 应用拥有者:输入员工号
- Marble版本号:2.0.0开始支持Netty
2、应用下添加运行的服务器(可添加多台)
- 服务器组:默认DEFAULT
- 服务器name:组内唯一
- 服务器描述
3、应用下添加Scheduler
- Scheduler名称:唯一Scheduler
- Scheduler 描述
- 服务器相关信息(点击选中输入打开的端口号)
4、Scheduler下添加多个Job
- Job名称:唯一job
- job 描述
- Cron表达式: 通过控件选择(具体含义,请 自行百度Cron表达式)
- Misfire策略选择(目前提供三种方式)
- Job执行参数:String类型
Marble的核心其实有两个部分:
- 1)Job的调度;
- 2)与服务端的通讯;
Marble的job调度使用了Quartz – 一款流行的开源作业调度框架。
与Job服务端的通讯使用了Apache Thrift / Netty(2.0.0开始) - 一款开源的高效RPC框架。
Marble在两个开源框架基础上做了必要的封装和针对公司使用的定制化修改。
### Marble实现原理 - 一次JOB调度过程调度服务与job服务交互:
- 服务端(Job执行服务器)随spring启动打开Socket端口;
- OFFLINE配置执行频率后,到达job执行时间,通过SOCKET与job服务器建立连接;
- 调用job服务器的execute方法(不等待返回),关闭本次socket连接;
TODO
### Marble实现原理 - Misfire策略#Marble优势
- 方便的第三方透明接入:应用接入Marble只需要三步;
- 灵活的JOB配置:支持同步JOB和异步JOB,提供Cron表达式可视编辑插件进行频率配置;
- 集群环境支持:集群下单机器执行;
- 系统容错:动态监测集群状态避开集群中异常状态机器;
- 调度过程全程追踪:每次调度都唯一标志流水号,log进行调度前、调度中、调度后的日志记录
#Marble存在的问题 + 后续计划
- 由于集群依赖于DB,如果大量JOB管理会造成DB的资源竞争加剧,从而引发JOB的misfire;
- JOB的执行是单机不能发挥集群的优势;
- 更多功能等待添加,比如:监控,报警,分析等;
- 暂时不支持执行中JOB的暂停:目前Marble只能暂停调度,不能暂停执行中的JOB;
- ...
TODO
# 关于作者 Java菜鸟。联系方式[email protected] or [email protected]