在现实生活中,会出现这样的例子,比如烧水,一般烧了20分钟后,水开了,会需要及时换水,再烧20分钟,水又开了,继续提醒,比如上班,每天早晨8点钟的闹钟会及时提醒, 那么在java程序中如何实现 这种 已经被设定了的定时的任务呢,在下面会依次介绍关于如何实现定时器来操作一些定时任务的业务需求。
当前java程序中 能够实现定时的 主要有 三种 方式 ,分别是: java定时器 , spring定时器 , quartz定时器.下面依次讲讲他们的应用!
<1>java定时器的应用。
其实java很早就有解决定时器任务的方法了,java提供了了类
java.util.TimerTask类基于线程的方式来实现定时任务的操作,然后再提供java.util.Timer类来注册调用,先创建一个类
RingTask
继承
java.util.TimerTask,实现run方法,相关代码如下:
package timer;
import java.util.TimerTask;
/**
* 这是一个打铃的程序,必须隔一段时间打一次
* @author sam
*
*/
public class RingTask extends TimerTask{
public RingTask() {
// TODO Auto-generated constructor stub
}
public RingTask(int s,int d) {
// TODO Auto-generated constructor stub
this.second = s;
this.delay = d;
}
int second = 1;
int delay = 1;
public void setSecond(int second) {
this.second = second;
}
public void setDelay(int delay) {
this.delay = delay;
}
@Override
public void run() {
// TODO Auto-generated method stub
System.out.println("我是打铃程序!"+"我第一次打铃延迟了"+delay+"秒!");
System.out.println("打铃了!每过"+second+"秒一次");
}
}
定义好后,下面需要注册调用了,注册调用的方法如下:
public static void main(String[] args) {
//以 java定时器的模式调用
Timer timer = new Timer();
timer.schedule(
new RingTask(3,3), //需要注册的定时类
3000, //最开始先延迟3秒的时间
3000); //每隔3秒的时间调用一次
}
一个简单的java定时器就写好了,方便而简介,但是有不好的
缺点: 如果需要实现每天早晨7点钟的定时执行一次,且周末的时候早晨7点钟不需要提醒,那这个可就不够用了,并且如果需要服务器一开启就触发这个定时器,则这种注册调用的方法也是不行的。
<2>Spring定时器的应用。
spring定时器是在spring框架中应用较成熟的一种方式,spring将定时任务的调用部分提到了配置文件当中,使定时器的触发条件变得更加灵活,spring定时器的实现,仍然需要 继承
java.util.TimerTask,实现run方法 ,示例类上面已给出,调用的配置如下:
<!-- 定时器的配置 (spring定时器)-->
<!-- 要调度的bean配置 -->
<bean id="ringTask" class="timer.RingTask">
<!-- 给 属性 second 赋值 为 3 -->
<property name="second" >
<value>3</value>
</property>
<!-- 给 属性 delay 赋值 为 3 -->
<property name="delay" >
<value>3</value>
</property>
</bean>
<!--配置一个触发器 配置触发器的参数-->
<bean id="scheduleRingTask" class="org.springframework.scheduling.timer.ScheduledTimerTask">
<property name="delay" value="3000"></property> <!--第一次延迟3秒的时间-->
<property name="period" value="3000"></property>
<!--每隔3秒的时间执行一次-->
<property name="timerTask" ref="ringTask"></property>
<!--制定触发的类-->
</bean>
<!-- 总调度,用于启动定时器 -->
<bean id="timerFactory" class="org.springframework.scheduling.timer.TimerFactoryBean">
<property name="scheduledTimerTasks">
<list>
<ref bean="scheduleRingTask"/>
</list>
</property>
</bean>
在调用方面是不是灵活些了,且能够实现服务器已启动,就将定时器的执行纳入的被监控的范围,符合条件马上触发执行。但是还是存在
缺点: 对于指定了具体的年月日时分秒而执行的任务还是不能解决。
<3>Quartz定时器
Quartz是基于Spring框架之上的更加强大的定时器,它不仅可以轻松的实现前面两种定时器的功能,还实现了非常繁复的时间触发执行的任务,Quartz有两种方式来调度定时任务,
一是使用Spring提供的 MethodInvokingJobDetailFactoryBean 代理类,Quartz通过该代理类直接调度任务类的某个函数;二是任务类继承
QuartzJobBean类
或者实现org.quartz.Job接口,Quartz通过该父类或者接口进行调度。
先来看看实现前面个两种定时器的功能,现在先来举个例子,比如烧水,每1小时烧开一次,进行定时提醒,然后重新换水,再烧,。。。。
具体的烧水定时类代码如下:
package timer;
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.scheduling.quartz.QuartzJobBean;
/**
* 这是一个热水的程序,必要要经过一段时间烧热了,才提醒重新换水
* @author sam
* 这个类不管是继承 QuartzJobBean还是实现org.quartz.Job都行
*/
public class HotWaterTask extends QuartzJobBean
/*implements Job*/{
//
public void execute(JobExecutionContext arg0)
//
throws JobExecutionException {
//
// TODO Auto-generated method stub
//
System.out.println("我是热水程序,我第一烧水需要1小时");
//
System.out.println("水现在烧开了,要及时换水哦!");
//
}
@Override
protected void executeInternal(JobExecutionContext arg0)
throws JobExecutionException {
// TODO Auto-generated method stub
System.out.println("我是热水程序,
我第一烧水需要1小时 ");
System.out.println("水现在烧开了,要及时换水哦!");
}
}
类定义好了,下面需要配置进去,配置的代码如下:
<!-- 配置需要调度的任务类 -->
<bean id="hotWaterTask" class="org.springframework.scheduling.quartz.JobDetailBean">
<property name="jobClass" value="timer.HotWaterTask"></property>
</bean>
<!-- 配置一个触发器 -->
<bean id="hotWaterTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">
<property name="jobDetail" ref="hotWaterTask"></property>
<property name="startDelay" value="3600000"></property>
<property name="repeatInterval" value="
3600000"></property>
</bean>
<!-- 总调度,用于启动定时器 -->
<bean id="schedulerFactory" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers" >
<list>
<ref bean="hotWaterTrigger"/>
</list>
</property>
</bean>
看到了吗,在这里我们并没有直接声明一个 HotWaterTask Bean,而是声明了一个JobDetailBean。这个是Quartz的特点。JobDetailBean是Quartz的org.quartz.JobDetail的子类,它要求通过jobClass属性来设置一个Job对象。
好了,上面的任务已经实现了,下面看看 如何实现 具体的年月日时分秒执行的代码
Quartz在指定的时间执行 (很强大的代理定时执行机制)
(1) 定义上班闹钟定时类代码如下:
package timer;
/**
* 开始上班,这个程序要求每天(非周末)早晨八点需要启动一次
* @author sam
*
*/
public class StartWorkJob {
public void startWork(){
System.out.println("我是上班程序,每天(非周末)早晨八点需要启动一次");
System.out.println("上班了!~");
}
}
看到了吗,这个类
StartWorkJob
并没有继承任何类也没有实现任何接口,且方法
start
Work也是自己定义的,原有的业务代码不需要做任何更改。下面就要提到Quartz实现的一种机制,通过Spring提供的代理类(
MethodInvokingJobDetailFactoryBean
)来实现定时任务,这个类只需要提供它要代理的类以及要代理的方法,就能够很好的就行定时监控了,强大吧,相关的代码如下:
<!-- 配置需要定时的bean类 -->
<bean id="startWorkJob" class="timer.StartWorkJob"></bean>
<!-- 配置任务的具体类和方法 -->
<bean id="startWorkTask" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<!-- 要调用的bean -->
<property name="targetObject" ref="startWorkJob"></property>
<!-- 要调用的Method -->
<property name="targetMethod" value="startWork"></property>
<!-- 是否并发,false表示 如果发生错误也不影响下一次的调用 -->
<property name="concurrent" value="false"></property>
</bean>
<!-- 配置一个触发器 -->
<bean id="startWorkTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail" ref="startWorkTask"></property>
<property name="
cronExpression" value="0 * 13 * * ?"></property> <!--每天的下午1点的每分钟的0秒都执行一次-->
</bean>
<!-- 总调度,用于启动定时器 -->
<bean id="schedulerFactory" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers" >
<list>
<ref bean="startWorkTrigger"/>
</list>
</property>
</bean>
好了一个指定了具体时间的定时触发任务也已经实现了,下面来看看
cronExpression 有哪些需要知道的配置信息,信息如下:
一个cron表达式有至少6个(也可能7个)有空格分隔的时间元素。从左到右:
1.秒2.分3.小时4.月份中的日期(1-31)5.月份(1-12或JAN-DEC)6.星期中的日期(1-7或SUN-SAT)7.年份(1970-2099)
每个元素都显示的规定一个值(如6),一个区间(9-12),一个列表(9,11,13)或一个通配符(*)。因为4和6这两个元素是互斥的,因此应该通过设置一个问号(?)来表明不想设置的那个字段,“/”如果值组合就表示重复次数(10/6表示每10秒重复6次)。
示例如下:
0 0 12 * * ?---------------在每天中午12:00触发
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:00至2:59之间每分钟触发一次
0 0/5 14 * * ?---------------每天在下午2:00至2:59之间每5分钟触发一次
0 0/5 14,18 * * ?---------------每天在下午2:00至2:59和6:00至6:59之间的每5分钟触发一次
0 0-5 14 * * ?---------------每天在下午2:00至2:05之间每分钟触发一次
0 10,44 14 ? 3 WED---------------每三月份的星期三在下午2:00和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, 2003, 2004 and 2005年的每个月的最后一个星期五的上午10:15触发
0 15 10 ? * 6#3---------------在每个月的第三个星期五的上午10:15触发
0 0 12 1/5 * ?---------------从每月的第一天起每过5天的中午12:00时触发
CronTrigger
CronTriggers往往比SimpleTrigger更有用,如果您需要基于日历的概念,而非SimpleTrigger完全指定的时间间隔,复发的发射工作的时间表。
CronTrigger,你可以指定触发的时间表如“每星期五中午”,或“每个工作日9:30时”,甚至“每5分钟一班9:00和10:00逢星期一上午,星期三星期五“。
即便如此,SimpleTrigger一样,CronTrigger拥有的startTime指定的时间表时生效,指定的时间表时,应停止(可选)结束时间。
Cron Expressions
cron的表达式被用来配置CronTrigger实例。 cron的表达式是字符串,实际上是由七子表达式,描述个别细节的时间表。这些子表达式是分开的空白,代表:
- 1. Seconds
- 2. Minutes
- 3. Hours
- 4. Day-of-Month
- 5. Month
- 6. Day-of-Week
- 7. Year (可选字段)
例 "0 0 12 ? * WED" 在每星期三下午12:00 执行,
个别子表达式可以包含范围, 例如,在前面的例子里("WED")可以替换成 "MON-FRI", "MON, WED, FRI"甚至"MON-WED,SAT".
“*” 代表整个时间段.
每一个字段都有一套可以指定有效值,如
Seconds (秒) :可以用数字0-59 表示,
Minutes(分) :可以用数字0-59 表示,
Hours(时) :可以用数字0-23表示,
Day-of-Month(天) :可以用数字1-31 中的任一一个值,但要注意一些特别的月份
Month(月) :可以用0-11 或用字符串 “JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV and DEC” 表示
Day-of-Week(每周):可以用数字1-7表示(1 = 星期日)或用字符口串“SUN, MON, TUE, WED, THU, FRI and SAT”表示
“/”:为特别单位,表示为“每”如“0/15”表示每隔15分钟执行一次,“0”表示为从“0”分开始, “3/20”表示表示每隔20分钟执行一次,“3”表示从第3分钟开始执行
“?”:表示每月的某一天,或第周的某一天
“L”:用于每月,或每周,表示为每月的最后一天,或每个月的最后星期几如“6L”表示“每月的最后一个星期五”
“W”:表示为最近工作日,如“15W”放在每月(day-of-month)字段上表示为“到本月15日最近的工作日”
““#”:是用来指定“的”每月第n个工作日,例 在每周(day-of-week)这个字段中内容为"6#3" or "FRI#3" 则表示“每月第三个星期五”
1)Cron表达式的格式:秒 分 时 日 月 周 年(可选)。
字段名 允许的值 允许的特殊字符
秒 0-59 , - * /
分 0-59 , - * /
小时 0-23 , - * /
日 1-31 , - * ? / L W C
月 1-12 or JAN-DEC , - * /
周几 1-7 or SUN-SAT , - * ? / L C #
年 (可选字段) empty, 1970-2099 , - * /
“?”字符:表示不确定的值
“,”字符:指定数个值
“-”字符:指定一个值的范围
“/”字符:指定一个值的增加幅度。n/m表示从n开始,每次增加m
“L”字符:用在日表示一个月中的最后一天,用在周表示该月最后一个星期X
“W”字符:指定离给定日期最近的工作日(周一到周五)
“#”字符:表示该月第几个周X。6#3表示该月第3个周五
2)Cron表达式范例:
每隔5秒执行一次:*/5 * * * * ?
每隔1分钟执行一次:0 */1 * * * ?
每天23点执行一次:0 0 23 * * ?
每天凌晨1点执行一次:0 0 1 * * ?
每月1号凌晨1点执行一次:0 0 1 1 * ?
每月最后一天23点执行一次:0 0 23 L * ?
每周星期天凌晨1点实行一次:0 0 1 ? * L
在26分、29分、33分执行一次:0 26,29,33 * * * ?
每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?