Spring boot基于ScheduledFuture實(shí)現(xiàn)定時(shí)任務(wù)
一、 背景
接上一篇,完成存儲(chǔ)過(guò)程的動(dòng)態(tài)生成后,需要構(gòu)建定時(shí)任務(wù)執(zhí)行存儲(chǔ)過(guò)程
二、 環(huán)境
1.此隨筆內(nèi)容基于spring boot項(xiàng)目
2.數(shù)據(jù)庫(kù)為mysql 5.7.9版本
3.jdk 版本為1.8
三、 內(nèi)容
1、定義接口和接口參數(shù)bean;
1)在上一篇博客bean 的基礎(chǔ)上把接口配置參數(shù)bean修改一下,添加一個(gè)配置參數(shù)值和排序字段;在添加一個(gè)監(jiān)測(cè)項(xiàng)的bean,想查看其他的bean信息,請(qǐng)移步
@Entity
@Table(name="monitor_warn_item")
public class MonitorWarnItem {
@Id
private String id;
private String proName;//名稱
private String rule;
private String send_content;
private String recommend_value;// 建議值
private String standard_value; // 標(biāo)準(zhǔn)值
private Integer fre_num;
private String frequency;
private String status;
private String warnType;
private String warn_date_num;// 監(jiān)測(cè)頻次
//此處省略get、set…
}
@Entity
@Table(name="qt_interface_parameter")
public class QtInterfaceParameter {
@Id
private String id;
@Column(name="inter_id")
private String interId;
private String name; //參數(shù)名稱
private String explain_info; //參數(shù)描述
private String type;// 輸入輸出類型
private String paraType; // 參數(shù)類型
private Integer paraLen;
private Integer paraValue; // 參數(shù)值
private Integer order_num; // 排序字段
//此處省略get、set…
}
2、定義ScheduledFuture定時(shí)任務(wù)
1) 添加接口
public interface TestService {
ResultInfo initMonitor(String Id);<br> // 省略之前的...
}
2) 編寫實(shí)現(xiàn)類
@Service
public class TestServiceImpl implements TestService {
@Autowired
private MonitorWarnItemRepository monitorWarnItemRepository
@Autowired
private ThreadPoolTaskScheduler threadPoolTaskScheduler;
@Bean
public ThreadPoolTaskScheduler threadPoolTaskScheduler() {
return new ThreadPoolTaskScheduler();
}
List<Map<String, Object>> mapList = new ArrayList<Map<String, Object>>(); // 新建任務(wù)信息集合
/**
* 初始化監(jiān)測(cè)項(xiàng)
*
* @param Id
* @return
*/
@Override
@Transactional
public ResultInfo initMonitor(String Id) {
ResultInfo info = new ResultInfo();
String msg = "";
MonitorWarnItem item = monitorWarnItemRepository.findId(Id);
msg =buildTask(item);
info.setResult(1);
info.setMsg("初始化成功,初始化返回信息:" + msg);
System.out.println(msg);// 日志打印
return info;
}
/**
* 配置任務(wù)信息
*
* @param qt
* @return
*/
private String buildTask(MonitorWarnItem qt) {
String msg = "";
if (IsFure(qt.getId())) {
List<QtInterface> InterList = qtInterfaceRepository.QueryInterFaceByItemId(qt.getId());
if (InterList.size() > 0) {
Map<String, Object> map_future = new HashMap<>();
ScheduledFuture<?> future;// 監(jiān)測(cè)任務(wù)
List<QtInterfaceParameter> para = qtInterfaceParameterRepository.QueryInfoByInterId(InterList.get(0).getId()); // 查找參數(shù)信息
List<String> map = new ArrayList<>(para.size());
if (para.size() > 0) { // 參數(shù)集合
for (QtInterfaceParameter pa : para) {
for (int item = 1; item <= para.size(); item++) {
if (item == pa.getOrder_num()) { // 根據(jù)字段排序來(lái)設(shè)置參數(shù)值的順序
map.add(pa.getPara_value()); // 設(shè)置值
item++;
}
}
}
}
QuartzTaskService service = new QuartzTaskService(InterList.get(0).getName(), map, jdbcTemplate, qt);
if (!"".equals(qt.getWarn_date_num()) && qt.getWarn_date_num() != null) {
future = threadPoolTaskScheduler.schedule(service, new CronTrigger(qt.getWarn_date_num()));// 初始化任務(wù),第二個(gè)參數(shù)是Cron表達(dá)式
if (future != null) {
map_future.put("future", future);
map_future.put("id", InterList.get(0).getItemId());
map_future.put("status", "0");
mapList.add(map_future);
}
} else {
msg += " 監(jiān)測(cè)項(xiàng):" + qt.getProName() + " 監(jiān)測(cè)頻次字段為空,不能執(zhí)行計(jì)劃!";
}
} else {
msg += " 監(jiān)測(cè)項(xiàng):" + qt.getProName() + " 沒(méi)有查找到接口配置信息";
}
} else {
msg += " 監(jiān)測(cè)項(xiàng):" + qt.getProName() + " 已經(jīng)啟動(dòng),請(qǐng)不要重復(fù)啟動(dòng)。";
}
return msg;
}
}
3) 構(gòu)建任務(wù)處理線程類
public class QuartzTaskService implements Runnable {
private JdbcTemplate jdbcTemplate;
private String proName;
private List<String> maplist;
private MonitorWarnItem item;
public QuartzTaskService(String proName,List<String> maplist,JdbcTemplate jdbcTemplate ,MonitorWarnItem item){
this.proName=proName;
this.maplist=maplist;
this.jdbcTemplate=jdbcTemplate;
this.item=item;
}
protected void executeInternal() throws JobExecutionException {
SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
StringBuffer bf=new StringBuffer();
bf.append("call ");
bf.append(proName);
bf.append("(");
int i=1;
for(String map:maplist){
if(i==maplist.size()){ // 最后一位
bf.append("'"+map+"')");
}else {
bf.append("'" + map + "',");
}
i++;
}
jdbcTemplate.batchUpdate(bf.toString());
System.out.println("執(zhí)行了過(guò)程:" +proName+"當(dāng)前參數(shù)順序:"+bf.toString()+ " 當(dāng)前時(shí)間 "+ sdf.format(new Date()));
}
@Override
public void run() {
try {
executeInternal(); // 調(diào)用執(zhí)行
} catch (JobExecutionException e) {
e.printStackTrace();
}
}
4) 此處是用的List保存的任務(wù)信息,在項(xiàng)目重啟之后這個(gè)東西就沒(méi)了,也就是說(shuō)定時(shí)任務(wù)就全丟了,so,這里考慮使用數(shù)據(jù)庫(kù)來(lái)持久化保存調(diào)度任務(wù)信息, 或者在項(xiàng)目啟動(dòng)的時(shí)候?qū)懸粋€(gè)配置來(lái)調(diào)用啟動(dòng)定時(shí)任務(wù)
@Component
@Order(1)
public class StartTask implements CommandLineRunner {
@Autowired
private TestService testService;
public String setTask(){
Calendar cale = null;
cale = Calendar.getInstance();
int year = cale.get(Calendar.YEAR);
MonitorWarnItem itemList=testService.QueryByStatus ("1");// 根據(jù)狀態(tài)查詢需要啟動(dòng)的監(jiān)測(cè)項(xiàng)
if(itemList.size()>0){ // 存在需要啟動(dòng)的檢測(cè)項(xiàng)
For(MonitorWarnItem qt: itemList)
testService.initMonitor(qt);// 啟動(dòng)任務(wù)列表和消息
}
return "";
}
@Override
public void run(String... args) throws Exception {
setTask ();
}
}
5)最后附上一個(gè)我使用的返回處理類
public class ResultInfo<T> {
private Integer result;
private String msg;
private T rows;
private int total;
//省略其他處理
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- SpringBoot定時(shí)任務(wù)參數(shù)運(yùn)行代碼實(shí)例解析
- SpringBoot整合SpringTask實(shí)現(xiàn)定時(shí)任務(wù)的流程
- Spring Boot @Scheduled定時(shí)任務(wù)代碼實(shí)例解析
- SpringBoot集成Quartz實(shí)現(xiàn)定時(shí)任務(wù)的方法
- Spring Boot監(jiān)聽Redis Key失效事件實(shí)現(xiàn)定時(shí)任務(wù)的示例
- springboot實(shí)現(xiàn)多實(shí)例crontab搶占定時(shí)任務(wù)(實(shí)例代碼)
- SpringBoot基于數(shù)據(jù)庫(kù)的定時(shí)任務(wù)統(tǒng)一管理的實(shí)現(xiàn)
- Spring Boot定時(shí)任務(wù)單線程多線程實(shí)現(xiàn)代碼解析
相關(guān)文章
Spring Boot開發(fā)Web應(yīng)用詳解
這篇文章主要介紹了Spring Boot開發(fā)Web應(yīng)用詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
Java流程控制之循環(huán)結(jié)構(gòu)for,增強(qiáng)for循環(huán)
這篇文章主要介紹了Java流程控制之循環(huán)結(jié)構(gòu)for,增強(qiáng)for循環(huán),for循環(huán)是編程語(yǔ)言中一種循環(huán)語(yǔ)句,而循環(huán)語(yǔ)句由循環(huán)體及循環(huán)的判定條件兩部分組成,其表達(dá)式為:for(單次表達(dá)式;條件表達(dá)式;末尾循環(huán)體){中間循環(huán)體;},下面我們倆看看文章內(nèi)容的詳細(xì)介紹2021-12-12
Springboot Vue可配置調(diào)度任務(wù)實(shí)現(xiàn)示例詳解
這篇文章主要為大家介紹了Springboot Vue可配置調(diào)度任務(wù)實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-01-01
SpringCloud Gateway 路由配置定位原理分析
本節(jié)主要了解系統(tǒng)中的謂詞與配置的路由信息是如何進(jìn)行初始化關(guān)聯(lián)生成路由對(duì)象的。每個(gè)謂詞工廠中的Config對(duì)象又是如何被解析配置的2021-07-07
如何解決hibernate一對(duì)多注解懶加載失效問(wèn)題
這篇文章主要介紹了解決hibernate一對(duì)多注解懶加載失效的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
Springcloud seata nacos環(huán)境搭建過(guò)程圖解
這篇文章主要介紹了Springcloud seata nacos環(huán)境搭建過(guò)程圖解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
spring/springboot整合dubbo詳細(xì)教程
今天教大家如何使用spring/springboot整合dubbo,文中有非常詳細(xì)的圖文介紹及代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴有很好地幫助,需要的朋友可以參考下2021-05-05

