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

spring任务调度SchedulerFactoryBean、CronTriggerFactoryBean(简单,明了)[通俗易懂]http://img.blog.csdn.net/20160530121730543?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Centerok,在控制台打印一句话,每隔一秒打印一次: 两步得到上述效果:…

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

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

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

 

两步得到上述效果:

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

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

 

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(根据实际需求,自己命名)

 

 

<?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>

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

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

先看示列:”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

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

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

 

 

 

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

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

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

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

 

<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 

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

 

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

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

 

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

 

 

 

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

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

 

比如每隔30mins执行一次:

 

@Scheduled(cron = "0 */30 * * * ?")//暂时先设置成半小时一次

 

 

Spring的配置文件:

 

<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"/>

 

 

 

 

 

@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的倍数的时候开始执行的:

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

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

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

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • DDR中的ODT功能详解及波形对比[通俗易懂]

    DDR中的ODT功能详解及波形对比[通俗易懂]ODT(ondietermination)即为片内端接,就是将端接电阻放在了芯片内部,这个功能只有在DDR2以上的数据信号才有。而有了ODT功能,原本需要在PCB板上加串联电阻的数据信号就不需要再额外添加端接了,只需要芯片内部打开ODT的端接功能,且这个端接可调。以下就是ODT的端接情况,如图所示:当数据读操作的时候,主控芯片(CPU)读取内存颗粒的数据,此时主控为接收端,可根据需要选择是否打开ODT功能;当数据写操作的时候,主控芯片(CPU)将数据写入内存颗粒,此时颗粒为接收端,也可以根据需要

    2025年10月10日
    4
  • Linux Samba服务器搭建

    Linux Samba服务器搭建

    2021年8月6日
    58
  • android+制作开机动画,Android 开机动画制作详解

    android+制作开机动画,Android 开机动画制作详解Android开机动画作为一个经常被DIY固件的玩家熟悉的内容,我这里简单做个总结。一、Android的2种类型:原生动画和第三方动画。1.原生动画Android标准代码是带有一个开机动画的,就是我们常见的带有滚动阴影的Android字样,如下图为标准Android原生动画:2.第三方动画:Android支持客户自定义第三方动画,我们只要将做好的动画文件,命名为…

    2022年5月14日
    33
  • python读取pkl文件_pkl文件是怎么训练出来的

    python读取pkl文件_pkl文件是怎么训练出来的importosimportpickledefread_pickle(work_path):data_list=[]withopen(work_path,”rb”)asf:whileTrue:try:data=pickle.load(f)data_list.append(data)

    2025年10月16日
    2
  • Android布局优化之ViewStub、include、merge使用与源码分析

    Android布局优化之ViewStub、include、merge使用与源码分析在开发中UI布局是我们都会遇到的问题,随着UI越来越多,布局的重复性、复杂度也会随之增长。首先用得最多的应该是include,按照官方的意思,include就是为了解决重复定义相同布局的

    2022年6月28日
    30
  • FindWindow和FindWindowEx

    FindWindow和FindWindowEx函数型:HWNDFindWindow(LPCTSTRIpClassName,LPCTSTRIpWindowName);IpClassName:指向一个指定了类名的空结束字符串或一个标识类名字符串的成员的指针。如果该参数为一个成员,则它必须为前次调用theGlobaIAddAtom函数产生的全局成员。该成员为16位,必须位于lpClassName的低16位,高位必须为0。如果为NULL,

    2022年5月31日
    35

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号