定时框架quartz
- 1. 使用场景
- 2. corn表达式
- 3. 简单使用
- 4. 注意点
1. 使用场景
-
定时任务
周期性执行 (每隔多长时间)
定时执行(每天几点) -
系统之间的数据交换
两个系统之间的数据 -
数据整理
报表打印…
2. corn表达式
corn 表达式在线生成
秒 分钟 小时
0 0/30 0/1 * * ? 代表每天每30分钟运行一次0 0/3 0/1 * * ? 代表每天每3分钟运行一次0/1 0/1 0/1 * * ? 代表每天每1秒钟运行一次0 11 23 * * ? 代表每天晚上23:11运行一次
"0 0 12 * * ?" 每天中午12点触发"0 15 10 ? * *" 每天上午10:15触发"0 15 10 * * ?" 每天上午10:15触发"0 15 10 * * ? *" 每天上午10:15触发"0 15 10 * * ? 2005" 2005年的每天上午10:15触发"0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟触发"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发"0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发
"0 15 10 15 * ?" 每月15日上午10:15触发
"0 15 10 L * ?" 每月最后一日的上午10:15触发"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发每天早上6点
0 6 * * *每两个小时
0 */2 * * *晚上11点到早上7点之间每两个小时,早上八点
0 23-7/2,8 * * *每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点
0 11 4 * 1-31月1日早上4点
0 4 1 1 *秒 分??时???日??月 星期
0??30 11,3??????*??1-5??11:30 和3:300 45 23,11 * * ? 每天11:45和23:45触发
3. 简单使用
核心注解scheduled,里面写corn表达式,将bean丢到ioc中
- jar包,需要整合spring使用
这里以controller开刀, 方法是无参无返,注解方式
- 头命名
xmlns:task="http://www.springframework.org/schema/task"
http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.1.xsd
<!-- quartz注解驱动 --><task:annotation-driven/>
package com.test.action;import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Controller;@Controller
public class DemoAction {/*** 系统间的数据交换 (放到公共的地方法)* 数据整理* */boolean task1Flag = true;// 任务积压 很短@Scheduled(cron = "0/1 * * * * ? ")public void show() {if (task1Flag) {task1Flag = false;System.out.println("hhhhh to do sth..");task1Flag = true;}}}
启动服务器就会…
4. 注意点
quartz如果设置每秒执行一次,会造成任务积压(1s掉quartz接管的方法,里面service的执行可能能大于1s),服务器内存会炸。
解决,像上面搞一个标记flag。
但是,每个方法都不能加判断,标记,显得冗余。
所以使用aop,对controller中的quartz进行环绕。
package com.test.aspect;import java.lang.reflect.Field;import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.reflect.MethodSignature;/*** 防止定时任务积压 切面*/
public class TaskAspect {//环绕public Object doRound(ProceedingJoinPoint pjp){//返回结果Object returnObj = null;// Object[] args = pjp.getArgs();
// Class class1 = pjp.getClass();//目标对象Object target = pjp.getTarget();
// Object this1 = pjp.getThis();//获得目标方法名Signature sig = pjp.getSignature();MethodSignature msig = null;if (!(sig instanceof MethodSignature)) {throw new IllegalArgumentException("该注解只能用于方法");}msig = (MethodSignature) sig;//方法名称String methodName = msig.getName();//根据方法名称获得对应的属性 task1Flag, task2Flag... try {Field field = target.getClass().getDeclaredField(methodName+"Flag");//设置允许访问field.setAccessible(true);//获得属性值boolean flag = Boolean.parseBoolean(field.get(target).toString());if(flag){//将标记改成falsefield.set(target, false);//执行目标try {returnObj = pjp.proceed();} catch (Throwable e) {e.printStackTrace();}//将标记改成truefield.set(target, true);}else{System.out.println("目标正在执行......");}} catch (SecurityException e1) {e1.printStackTrace();} catch (NoSuchFieldException e1) {e1.printStackTrace();} catch (IllegalArgumentException e) {e.printStackTrace();} catch (IllegalAccessException e) {e.printStackTrace();} return returnObj;}}