我们有很多定时任务在任务工程中执行,但是如果我们不加以监控,很有可能定时任务死掉或者卡住我们都不知道。

所以需要做一个任务监控。监控任务当前执行的状态。

还是那样,先让定时任务启动起来,当前我们使用的是spring提供的schedule执行定时任务。注意,这个默认情况下是单线执行,就是说你有很多定时任务,你的时间执行很频繁,他只会给你排队执行,不会到时间立马执行。如果想要配置多个定时任务同时执行,那么只需要在你任务标签上加上@Async。一般情况下。不是忒紧急的,不需要加。加上后,你就需要给你的方法调用的数据做好异步同步控制了。具体调用,可参考前一篇文章。

ok,废话不多说,开始弄代码。

1、spring配置加上以下配置,实现任务调度与切面生效。

    <!-- 切面开关 -->
<aop:aspectj-autoproxy />
<!-- 定时器开关 -->
<task:annotation-driven />

2、任务信息实现:

    /**
* 提醒当天到期还款的还款人发短信
*/
@MyTask(taskNo="smsNotifyRepaymenterToday", taskDesc="当日到期的还款人发短信")
@Scheduled(cron = "${SmsTaskExecutor.smsNotifyRepaymenterToday}")
public void smsNotifyRepaymenterToday() {
String message = StatusContants.TASK_INFO_SMS_NOTIFY_REPAYMENTER_TODAY.getName();
String taskNo = StatusContants.TASK_INFO_SMS_NOTIFY_REPAYMENTER_TODAY.getIndex();
if (!taskService.taskDo(taskNo)){return ;}
try{
taskService.taskUpdate(taskNo, StatusContants.SYSTEM_TABLE_STATUS_1.getIndex()); logger.info(message + ControllerContants.MESSAGE_START);
// 计算发送时间范围
String notifyDateStr = dictionaryVoService.findDeductDate(); // 会计日期当天
String deductDateStart = notifyDateStr;
String deductDateEnd = notifyDateStr;
// 发送
phoneMsgService.notifyRepaymenterTodayBySms(deductDateStart, deductDateEnd);
logger.info(message + ControllerContants.MESSAGE_END);
}catch (Exception e) {
logger.error(message+"异常", e);
exceptionInfoService.insert(message+"异常", StatusContants.LOG_TYPE_EXCPETION.getIndex(), MyStringUtils.getExceptionDetail(e), getClass().getName()+".smsNotifyRepaymenter");
}finally{
taskService.taskUpdate(taskNo, StatusContants.SYSTEM_TABLE_STATUS_0.getIndex());
}
}

3、注解标签实现:

@MyTask(taskNo="smsNotifyRepaymenterToday", taskDesc="当日到期的还款人发短信")
/**
* * <p>
* 定时任务监控,用于方法上<br/>
* 在新建定时任务时,在方法上写taskNo=任务编号,此时拦截器会根据编号信息,获取任务执行情况,更改任务情况,并且给任务加上异常捕获,
* 监控任务信息。
* </p>
* @author chenweixian 陈惟鲜
*/
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyTask { String taskNo(); String taskDesc();
}

4、监控切面的实现。

package com.chenweixian.task.aop;

import java.math.BigDecimal;
import java.util.Date; import net.sf.json.JSONObject; import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.annotation.Order;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component; import com.opensymphony.oscache.util.StringUtil; /**Spring的定时任务推送监控
*
* @author chenweixian
*
*/
@Aspect
@Order(1) // 排序
@Component
public class ScheduleMonitoringInterceptor {
private Logger logger = Logger.getLogger(getClass()); @Autowired
private ActiveMqUtil activeMqUtil; @Async
@Pointcut("@annotation(org.springframework.scheduling.annotation.Scheduled)") // 用于拦截标签
private void anyMethod(){}//定义一个切入点 /**
*
* @Title:doAround
* @Description: 环绕触发
* @author 陈惟鲜
* @date 2014年11月3日 下午1:58:45
* @param point
* @return
* @throws Throwable
*/
@Around("anyMethod() && @annotation(myTask)")
public Object doAround(ProceedingJoinPoint point, MyTask myTask) throws Throwable {
Date startTime = MyDateUtil.getCurrentDateTime();// 开始时间 // 执行完方法的返回值:调用proceed()方法,就会触发切入点方法执行
Object result = point.proceed();// result的值就是被拦截方法的返回值 Date endTime = MyDateUtil.getCurrentDateTime();// 结束时间
try{
this.writeLog(point, myTask, startTime, endTime);
}catch(Exception e){
logger.info("记录监控日志异常。。");
} return result;
} /***记录日志*/
private void writeLog(JoinPoint point, MyTask myTask, Date startTime, Date endTime){
String taskdesc = StatusContants.getName(StatusTypeContants.TYPE_ID_TASK_INFO, myTask.taskNo());
if (StringUtil.isEmpty(taskdesc)){
taskdesc = "";
} ProcedureMonitor procedureMonitor = new ProcedureMonitor();
procedureMonitor.setAutoId(UUIDUtils.generateUUID64());
procedureMonitor.setIpAddress(MyStringUtils.getIpString());// 机器IP
procedureMonitor.setStartTime(startTime);// 开始执行时间
procedureMonitor.setEndTime(endTime);// 结束执行时间
procedureMonitor.setConsumeTimes(new BigDecimal(endTime.getTime()-startTime.getTime()));// 总耗时时间
procedureMonitor.setExceptionFlag(StatusContants.SYSTEM_TABLE_STATUS_0.getIndex());// 是否异常
procedureMonitor.setTaskNo(myTask.taskNo()); // 任务编号
procedureMonitor.setTaskName(myTask.taskDesc()); // 任务编号对应名称
procedureMonitor.setRemark("【"+point.getSignature().getDeclaringTypeName()+"."+point.getSignature().getName()+"】"+taskdesc); // 所在的类.方法
procedureMonitor.setBusinessType(StatusContants.PROCEDURE_MONITOR_TASK.getIndex()); // 监控类型
// 发送mq消息
activeMqUtil.sendMq(ActivityMqContants.TASK_PROCEDURE_MONITOR, JSONObject.fromObject(procedureMonitor).toString());
}
}

5、完成。。

springAop Schedule,注解annotation 实现任务监控的更多相关文章

  1. springAop,注解annotation + redis 实现分布式锁

    当前流行的系统,就是分布式系统.所谓分布式,我个人理解,是很多的服务分布在不同的机器上,都是相同功能模块.但是容易出现一个问题,就是并发时的问题. 我们传统的锁,只能锁住一个服务器上的方法,让其在一个 ...

  2. Java的注解(Annotation)

    1.什么是注解 Annotation is code about the code, that is metadata about the program itself. Java注解,是Java5. ...

  3. Java Android 注解(Annotation) 及几个常用开源项目注解原理简析

    不少开源库(ButterKnife.Retrofit.ActiveAndroid等等)都用到了注解的方式来简化代码提高开发效率. 本文简单介绍下 Annotation 示例.概念及作用.分类.自定义. ...

  4. Java注解Annotation学习

    学习注解Annotation的原理,这篇讲的不错:http://blog.csdn.net/lylwo317/article/details/52163304 先自定义一个运行时注解 @Target( ...

  5. Java中注解Annotation的定义、使用、解析

    此例子,用于说明如何在Java中对“注解 Annotation”的定义.使用和解析的操作.注解一般用于自定义开发框架中,至于为什么使用,此处不作过多说明,这里只说明如何使用,以作备记.下面例子已测试, ...

  6. java基础-注解Annotation原理和用法

    在很多java代码中都可以看到诸如@Override.@Deprecated.@SuppressWarnings这样的字符,这些就是注解Annotation.注解最早在jdk5中被引入,现在已经成为j ...

  7. Enum枚举类|注解Annotation

    Enum枚举类 ①枚举类和普通类的差别: 使用 enum 定义的枚举类默认继承了 java.lang.Enum 类 枚举类的构造器仅仅能使用 private 訪问控制符 枚举类的全部实例必须在枚举类中 ...

  8. JUnit扩展:引入新注解Annotation

    发现问题 JUnit提供了Test Suite来帮助我们组织case,还提供了Category来帮助我们来给建立大的Test Set,比如BAT,MAT, Full Testing. 那么什么情况下, ...

  9. Java - 注解 (Annotation)

    Java - 注解 (Annotation)   一.基本的 Annotation     > 使用 Annotation 时要在其前面增加 @符号,并把该 Annotation 当成一个修饰符 ...

随机推荐

  1. ARTS 第十周打卡

    Algorithm : 做一个 leetcode 的算法题 编写一个函数来查找字符串数组中的最长公共前缀. 如果不存在公共前缀,返回空字符串 "". 示例 1: 输入: [&quo ...

  2. ORACLE 的前后台进程

    关于oracle用户进程,服务进程,后台进程 用户进程(User Process) 是一个需要与Oracle Server交互的程序 运行于客户端 当用户运行某个工具或应用程序(如SQL*Plus)时 ...

  3. Linux驱动函数解读

    一.kmalloc().kzalloc()和vmalloc() 这三个函数都可以分配连续的虚拟内存 除此之外,这三个函数的区别有: 1. kmalloc()和kzalloc()函数分配的物理内存也是连 ...

  4. Vue CLI 3开发中屏蔽的EsLint错误 (.eslintrc.js 在vue3+中 修改这个)

    1.关闭eslint校验有了eslint的校验,可以来规范开发人员的代码,是挺好的.但是有些像缩进.空格.空白行之类的规范,在开发过程中一直报错,未免太过于苛刻了.所以,我还是会选择关闭eslint校 ...

  5. c# asp.net 实现分页(pager)功能

    分页PagerHelper辅助类 using System;using System.Web; public class PagerHelper { #region 获取分页的Html代码 /// & ...

  6. C语言之反汇编揭秘

    title: 'C语言之反汇编揭秘' tags: 汇编与反汇编 categories: 汇编与反汇编 copyright: true abbrlink: 'b1c9' date: 2019-09-07 ...

  7. Django-redis配置cache和session

    CACHES = { "default": { "BACKEND": "django_redis.cache.RedisCache", &q ...

  8. SQL Server系统函数:元数据函数

    原文:SQL Server系统函数:元数据函数 1.列的长度.列名 --列的长度 select COL_LENGTH('dbo.wct', --表名 'wcid') --列名 --列名 select ...

  9. internal关键字

    internal修饰符可以用于类型或成员,使用该修饰符声明的类型或成员只能在同一程集内访问,接口的成员不能使用internal修饰符. 就是使用internal的类只能在同一个项目中使用,不能在别的项 ...

  10. Java Web 深入分析(7) Jetty原理解析

    1Jetty的基本架构 Jetty有一个基本的数据模型,这个模式就是handle,所有拷贝拓展的组件都被当做一个handler被添加到server中,然后由jetty统一管理. 1.1Jetty基本架 ...