spring任务调度SchedulerFactoryBean、CronTriggerFactoryBean(简单,明了)[通俗易懂]

2022-08-31 20:46:51 浏览数 (1)

大家好,又见面了,我是你们的朋友全栈君。

https://img-blog.csdn.net/20160530121730543?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Centerok,在控制台打印一句话,每隔一秒打印一次:

两步得到上述效果:

第一步:写一个测试类MyTimeTask继承TimerTask,并实现它的run()方法;

run主体打印一句话:System.out.println(“hello:” df.format(new Date()));

代码语言:javascript复制
package cn.com.jsoft.task;

import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.TimerTask;

public class MyTimeTask  extends TimerTask{
	SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

	@Override
	public void run() {
		// TODO Auto-generated method stub
		System.out.println("hello:" df.format(new Date()));
	}
	
	public static void main(String[] args) {
		/*SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
		System.out.println(df.format(new Date()));*/
	}

}

第二步:配置applicationContext-scheduler.xml(根据实际需求,自己命名)

代码语言:javascript复制
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:aop="http://www.springframework.org/schema/aop"
	xmlns:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation=" 
    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd 
    http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd 
    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
    
	<bean id="myTimeTask" class="cn.com.jsoft.task.MyTimeTask"/>
	
	<bean id="myTimeTaskJobDetail" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
		<property name="targetObject">
			<ref bean="myTimeTask" />
		</property>
		<property name="targetMethod">
			<value>run</value>
		</property>
		<property name="concurrent">
			<value>false</value>
		</property>
	</bean>
	<bean id="myTimeTaskTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean"><!--如果项目报错,把CronTriggerBean换成CronTriggerFactoryBean -->
		<property name="jobDetail">
			<ref local="myTimeTaskJobDetail" />
		</property>
		<property name="cronExpression">
			<!-- 每隔一分钟执行一次 -->
			<!-- 0 */1 * * * ? -->
			<value>*/1 * * * * ?</value>
		</property>
	</bean>
	
	<bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean" lazy-init="false" autowire="no">
		<property name="triggers">
			<list>
				<ref local="myTimeTaskTrigger" />
			</list>
		</property>
		<property name="autoStartup">
			<value>true</value>
		</property>
	</bean>
</beans>

说明:cronExpression参数配置(注意空格)

在线版:http://www.cronmaker.com/

<!– 每隔一秒钟执行一次 –>

<value>*/1 * * * * ?</value>

<!– 每隔一分钟执行一次 –>

<value>* */1 * * * ?</value>

从左到右分别是:秒,分,时,月的某天,月,星期的某天,年;其中年不是必须的,也就是说任何一个表达式最少需要六项。

先看示列:”0 0/30 8-10 5,20 * ?” 表示“每个月的5日和20日的8:00,8:30,9:00,9:30,10:00,10:30” 字符解释: ,:与,表式”,”两边的值都是需要执行的时间,如上例“5,20”,每个月的5日与20日。 –:表示值的范围,如上例“8-10”,从8点开始到10结束,包括8点与10点。 *:表式任意可合法的值,如上例“*”是处于月份的字段,所以代表1-12中的任意值,所以上例是指“每个月”。 /:增量,如上例是指从0分开始,每过30分钟取一次值。如果换成“5/8”就是从第5钟开始每过8分钟取一次值:8:05,8:13,8:21,8:29等等 ?:不指定值,就是“我也不知道”的意思,只能出现在“月的某天,星期的某天”项中。在什么情况下用呢?如上例如果指定值为星期一,那么可能会出现如4月5日不是星期一,这里就是不对应,有冲突,所以指定为”?”,也就是说我也不知道是星期几,只要是5日与20日就行了,至于是星期几我才不管呢! L:最后的,last的意思,只能出现在“月的某天,星期的某天”项中。表示当前月或当前星期的最后一天,注意的是星期的最后一天为星期六。 W:月中最接近指定日期的普通日(星期一到星期五),只能出现在“月的某天”,如”15W”就是说当前月最接近15日的普通日,如果当月的15是星期三就是星期三,如果当月的15是星期六那么就是昨天也就是星期五,如果当月的15是星期天则为第二天也就是星期一。 #:当前月的第N个星期X日,只能出现在“星期的某天”项中。如“6#3”就是说当前月的第三个星期五,注意”1-7″,1=星期天,2=星期一 等等。

(部分内容来自互联网)

  • “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触发

对“concurrent”设置false的深入剖析:请看下面的例子与解答,来自http://www.iteye.com/problems/54435

========================================华丽的分隔符=================================================

今天重新照这个写的时候发现几个问题:

①好像SpringMVC-scheduler.xml需要配置到web.xml文件中,这没有说明,我这里补充下!

建议命名的时候取统一的前缀,比如楼主的:SpringMVC-scheduler.xml,SpringMVC-DB.xml,然后在web.xml中引用的时候:

代码语言:javascript复制
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:SpringMVC-*.xml</param-value>
  </context-param>

②直接照着上面的代码复制粘贴,发现项目报错了:

Caused by: java.lang.IncompatibleClassChangeError: class org.springframework.scheduling.quartz.CronTriggerBean has

interface org.quartz.CronTrigger as super class

这个问题的解决办法是:把配置文件SpringMVC-scheduler.xml中的CronTriggerBean更换成CronTriggerFactoryBean,问题就解决了!

来自:http://www.cnblogs.com/interdrp/p/3587221.html

补充于:2016年5月30日 12:20:40

========================================华丽的分隔符=================================================

之前都是把配置放到xml里面,现在也可以使用注解的方式来跑定时任务了,一般而言不是经常变来变去的配置都可以使用注解的方式来操作quartz;

比如每隔30mins执行一次:

代码语言:javascript复制
@Scheduled(cron = "0 */30 * * * ?")//暂时先设置成半小时一次

Spring的配置文件:

代码语言:javascript复制
<description>Spring Configuration</description>
	
	<context:component-scan base-package="scheduler"/> 
	
	<!-- 配置任务线性池 -->
    <task:executor  id="executor" pool-size="10" /> 
    <task:scheduler id="scheduler" pool-size="10"/>
    <task:annotation-driven scheduler="scheduler" executor="executor" proxy-target-class="true"/>
代码语言:javascript复制
@component (把普通pojo实例化到spring容器中,相当于配置文件中的<bean id="" class=""/>)

proxy-target-class=”true”

该属性值默认为false,表示使用JDK动态代理织入增强;当值为true时,表示使用CGLib动态代理织入增强;但是,即使设置为false,如果目标类没有生命接口,则spring将自动使用CGLib动态代理.(以上来自:Spring3.X企业应用开发实战 P229)

通俗理解:

当要使用实现了某个接口的类让Spring来生成bean时,无需在aop配置中添加proxy-target-class,因为它默认为false.

但如果要使用一个指定的类,让Spring来生成bean,并使用它的某个方法时,需要在aop配置上加上一句proxy-target-class=”true”,否则用JUnit时,会出现:

Java.lang.ClassCastException: com.sun.proxy.$Proxy6 cannot be cast to glut.daoImp2.DAOImp2

类似的错误.

0 */30 * * * ?和0 0/30 * * * ? 都是指的是30的倍数的时候开始执行,有的时候容易误解*/30是指,从启动开始每隔30mins执行一次,这里是个坑,哈哈;

为了验证,我用30s来试试,确实证实了都是30的倍数的时候开始执行的:

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/142234.html原文链接:https://javaforall.cn

0 人点赞