前端定时器表达式可参考: layui 定时器表达式 插件 代码.
一、定时器控制表
CREATE TABLE `scheduled` (
`ID` bigint(20) NOT NULL AUTO_INCREMENT,
`CODE` varchar(50) DEFAULT NULL COMMENT '定时器编号',
`CORN` varchar(50) DEFAULT NULL COMMENT '定时器表达式',
`CORNDESC` varchar(200) DEFAULT NULL COMMENT '定时器表达式内容',
`CORNNAME` varchar(100) DEFAULT NULL COMMENT '定时器名称',
`CORNLINK` varchar(200) DEFAULT NULL COMMENT '定时接口 链接 eg:http://project/test?id=123456',
`METHOD` varchar(10) DEFAULT NULL COMMENT '请求方式GET/POST',
`STATUS` varchar(2) DEFAULT NULL COMMENT '状态1=启动 2=停止',
`STARTTIME` datetime DEFAULT NULL COMMENT '最后一次启动时间',
`ENDTIME` datetime DEFAULT NULL COMMENT '最后一次停止时间',
`REMARK` varchar(200) DEFAULT NULL COMMENT '备注说明',
PRIMARY KEY (`ID`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=25 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='定时器控制表';
二、功能实现代码
1.定时调度配置类+启动项目时自动启动数据库中开启状态的调度
/**
* 描述:
* 定时调度配置类
* @author 闲走天涯
* @create 2020/9/3 11:15
*/
@Slf4j
@Component
public class ScheduledConfig implements ApplicationRunner {
@Autowired
ScheduledDao scheduledDao;
@Autowired
RestTemplate restTemplate;
@Override
public void run(ApplicationArguments args) throws Exception {
try {
List<ScheDuledVo> vos = scheduledDao.findStartingScheduled();
for (ScheDuledVo vo : vos) {
MultiValueMap<String,Object> multiValueMap = new LinkedMultiValueMap<>();
multiValueMap.add("code",vo.getCode());
Map result = restTemplate.postForObject("http://ALI-ASSIST-SERVICE/scheduled/startCron",multiValueMap,Map.class);
log.info("【启动 定时调度】接口={},结果={}", vo.getCornLink(), result);
}
}catch (Exception e){
log.error("【启动 定时调度】失败",e);
}
}
}
2.开启、关闭调度
/**
* 描述:
* java 定时调度 控制 执行代码端
*
* @author 闲走天涯
* @create 2020/9/2 11:23
*/
@Slf4j
@RestController
@RequestMapping("/scheduled")
public class ScheduledController {
@Autowired
private RestTemplate restTemplate;
@Autowired
private ScheduledDao scheduledDao;
@Autowired
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
Map<String, ScheduledFuture> map = new HashMap<>();
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
return new ThreadPoolTaskScheduler();
}
/**
* 启动定时器
*
* @param code 定时器编号
* @return
*/
@RequestMapping("/startCron")
public Map startCron(String code) {
Map result = new HashMap();
result.put("code", "1");
try {
log.info("【定时调度】启动 参数-》 code={}", code);
if (!VerifyData.strIsNotNull(code)) {
result.put("code", "1");
result.put("msg", "参数错误");
return result;
}
ScheDuledVo vo = scheduledDao.selectScheDuleByCode(code);
log.info("【定时调度】启动 参数-》 vo={}", vo.toString());
if (!VerifyData.strIsNotNull(vo.getCorn()) || !VerifyData.strIsNotNull(vo.getCode()) || !VerifyData.strIsNotNull(vo.getCornName()) || !VerifyData.strIsNotNull(vo.getCornLink()) || !VerifyData.strIsNotNull(vo.getMethod())) {
result.put("code", "1");
result.put("msg", "参数错误");
return result;
}
ScheduledFuture future = threadPoolTaskScheduler.schedule(new taskRunnable(vo), new CronTrigger(vo.getCorn()));
map.put(vo.getCode(), future);
log.info("【定时调度】启动 定时器编号={},名称={}", vo.getCode(), vo.getCornName());
if (!"1".equals(vo.getStatus())) {
int i = scheduledDao.updateScheduledStatus(code, "1");
}
} catch (Exception e) {
log.error("【定时调度】启动 失败", e);
result.put("code", "1");
result.put("msg", "系统错误");
return result;
}
result.put("code", "3");
result.put("msg", "启动成功");
return result;
}
/**
* 停止定时器
*
* @param code
* @return
*/
@RequestMapping("/stopCron")
public Map stopCron(String code) {
Map result = new HashMap();
result.put("code", "1");
try {
log.info("【定时调度】停止 参数-》 code={}", code);
if (!VerifyData.strIsNotNull(code)) {
result.put("code", "1");
result.put("msg", "参数错误");
return result;
}
ScheDuledVo vo = scheduledDao.selectScheDuleByCode(code);
log.info("【定时调度】停止 参数-》 vo={}", vo.toString());
if (!VerifyData.strIsNotNull(vo.getCorn()) || !VerifyData.strIsNotNull(vo.getCode()) || !VerifyData.strIsNotNull(vo.getCornName()) || !VerifyData.strIsNotNull(vo.getCornLink()) || !VerifyData.strIsNotNull(vo.getMethod())) {
result.put("code", "1");
result.put("msg", "参数错误");
return result;
}
ScheduledFuture future = map.get(vo.getCode());
if (future != null) {
future.cancel(true);
log.info("【定时调度】停止 定时器编号={},名称={}", vo.getCode(), vo.getCornName());
map.remove(vo.getCode());
if (!"2".equals(vo.getStatus())) {
int i = scheduledDao.updateScheduledStatus(code, "2");
}
} else {
result.put("code", "2");
result.put("msg", "查无定时器编号,错误");
return result;
}
} catch (Exception e) {
log.error("【定时调度】停止 失败", e);
result.put("code", "1");
result.put("msg", "系统错误");
return result;
}
result.put("code", "3");
result.put("msg", "停止成功");
return result;
}
/**
* 定时器执行方法
*
* @param vo
*/
private void dotask(ScheDuledVo vo) {
log.info("【定时任务】" + vo.getCode() + ":" + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
try {
JSONObject result = new JSONObject();
if ("get".equals(vo.getMethod().toLowerCase())) {
result = restTemplate.getForObject(vo.getCornLink(), JSONObject.class);
} else if ("post".equals(vo.getMethod().toLowerCase())) {
if (vo.getCornLink().indexOf("?") != -1) {
String link = vo.getCornLink().substring(0, vo.getCornLink().indexOf("?"));
MultiValueMap<String, Object> multiValueMap = new LinkedMultiValueMap<>();
String params = vo.getCornLink().substring(vo.getCornLink().indexOf("?") + 1, vo.getCornLink().length());
if (params.indexOf("&") != -1) {
String[] paramArr = params.split("&");
for (String param : paramArr) {
multiValueMap.add(param.split("=")[0], param.split("=")[1]);
}
} else {
multiValueMap.add(params.split("=")[0], params.split("=")[1]);
}
result = restTemplate.postForObject(link, multiValueMap, JSONObject.class);
} else {
result = restTemplate.postForObject(vo.getCornLink(), null, JSONObject.class);
}
}
//log.info("【定时任务】"+vo.getCode()+":result="+JSONObject.toJSONString(result));
} catch (Exception e) {
log.error("【定时任务】接口(" + vo.getCornLink() + ") 访问失败" + e.getMessage());
}
}
//多线程
class taskRunnable implements Runnable {
private ScheDuledVo vo;
public taskRunnable(ScheDuledVo vo) {
this.vo = vo;
}
@Override
public void run() {
//编写你自己的业务逻辑
dotask(vo);
}
}
}
三、vo类
@Data
public class ScheDuledVo {
private Long id;
private String corn;//定时器表达式
private String cornDesc;//定时器表达式内容
private String cornName;//定时器名称
private String status;//定时器状态 1=开启 2=关闭
private Date startTime;//上一次开始时间
private Date endTime;//上一次结束时间
private String remark;//备注说明
private String cornLink;//定时器访问接口链接 eg:http://ali-assist/deal/test?id=123456
private String method;//定时器访问接口方式 GET POST
private String code;//定时器编号
private String startTimeStr;//上一次开始时间
private String endTimeStr;//上一次结束时间
}