一:java Timer 在 java.util.Timer包中的Timer 类可以实现定时功能。
只有当任务的执行时间到来时,ScheduedExecutor 才会真正启动一个线程,其余时间 ScheduledExecutor 都
是在轮询任务的状态。此定时器在抛出运行时异常时 虽然不会影响其他启动的任务但是本次的任务将会退出,
自此后不再执行。
三:基于框架Spring 的定时器:
如果你用了Spring 框架了,那么 本人极力推荐这种定时器,因为他集齐上述两种方式的有点于一身,并且没
有前两者的缺点。即:当前任务中抛出了运行时异常,当前任务线程不会退出。如果程序是每天定时任务,那
么在下一个任务运行的时间点任务仍然继续。
使用Spring 定时器需要的jar包:quartz-all-1.6.0.jar 和 jta.jar两个jar包。
配置文件:
在Spring 的业务逻辑bean容器配置文件中添加如下配置:
<!-- Spring 定时器 start -->
<bean id="activitytask" class="com.gop.service.activitytimer.ActivityTimer">
<property name="activityServiceImpl">
<ref bean="activityServiceImpl"/>
</property>
<property name="activityRecordServiceImpl">
<ref bean="activityRecordServiceImpl"/>
</property>
<property name="mailsServiceImpl">
<ref bean="mailsServiceImpl"/>
</property>
</bean>
//如果是一个普通的类,不需要注入dao层
<bean id=" activitytask " class=" com.gop.service.activitytimer.ActivityTimer "></bean>
//上面的两个bean节点根据自己的情况任选一个,下面的bean节点都是需要的。
<bean id="triggerB" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail">
<ref bean="jobDetailB"/>
</property>
<property name="cronExpression">
<value>0 34 17 * * ?</value>
</property>
</bean>
<bean id="jobDetailB" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetObject">
<ref bean="activitytask"/>
</property>
<property name="targetMethod" value="activityStart"/> //----①
</bean>
<bean id="scheduler" lazy-init="false" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers">
<list>
<ref bean="triggerB"/>
</list>
</property>
</bean>
实现类: com.gop.service.activitytimer.ActivityTimer
public class ActivityTimer {
public static final SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
public static final SimpleDateFormat sdfTim=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
private ActivityServiceImpl activityServiceImpl;
private MailsServiceImpl mailsServiceImpl;
private ActivityRecordServiceImpl activityRecordServiceImpl;
public ActivityTimer(){}
/**
* 定时器主处理方法
*/
public void activityStart() { //这里的方法名与配置文件中 ① 处名称一致
//每天晚24点 处理线上生效的活动.给玩家发奖。
// activityManage 为自定义处理逻辑方法。
activityManage();
}
}
时间的配置如下:
<value> 0 34 17 * * ?< value> 代表每天的17:34:00 开始执行任务
时间大小由小到大排列,从秒开始,顺序为 秒,分,时,天,月,年 *为任意 ?为无限制。由此上面所配置的内容就是,在每天的16点26分启动znrwdb方法
具体时间设定可参考
"0/10 * * * * ?" 每10秒触发
"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触发
每隔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 * * ?
1. 使用 Timer 进行任务调度
<span style="font-size:18px;">public class TimerTest extends TimerTask { public TimerTest(){} </span>
<span style="font-size:18px;">@Override public void run() { System.out.println("定时任务处理逻辑"); } </span>
<span style="font-size:18px;">public static void main(String[] args) { Timer timer = new Timer(); long delay1 = 1 * 1000; long period1 = 1000; // 从现在开始延时 1 秒钟之后开始,每隔 1 秒钟执行一次run 方法。 timer.schedule(new TimerTest("job1"), delay1, period1); long delay2 = 2 * 1000; long period2 = 2000; </span><pre name="code" class="displaycode"><span style="font-size:18px;"> // 从现在开始延时 2秒钟之后开始,每隔 2 秒钟执行一次run 方法。 timer.schedule(new TimerTest(), delay2, period2); //指定每天的指定时间执行 定时任务(run 方法的逻辑处理) SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd "); String nowDate = sdf1.format(new Date()); String timerTime = "23:49:00"; Timer timer = new Timer(); String timerDateStr = nowDate + timerTime; Date timerDate = null; try { timerDate = sdf.parse(timerDateStr); } catch (ParseException e) { e.printStackTrace(); } //</span><span style="font-size:18px;">timerDate时间小于当前时间则自动执行一次业务逻辑,当前时间未达到指定该参数的时间时, //只有达到该参数指定时间时才会执行任务 </span><span style="font-size:18px;"> timer.schedule(new </span><span style="font-size:18px;">TimerTest()</span>,<span style="font-size:18px;"> timerDate);</span> } <span style="font-size:18px;">优缺:Timer 的优点在于简单易用,但由于所有任务都是由同一个线程来调度,因此所有任务都是串行执行的, 同一时间只能有一个任务在执行,前一个任务的延迟或异常都将会影响到之后的任务。而且当前任务如果抛出 运行时异常则该任务终止,如果<span style="color:#FF0000;">当前任务是每天定时任务</span>则下次时间点的任务则不会执行。</span> <span style="font-size:18px;"> <strong>二:ScheduledExecutor</strong></span> <pre name="code" class="displaycode"><span style="font-size:18px;">import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.TimeUnit; public class ScheduledExecutorTest implements Runnable { private String jobName = ""; public ScheduledExecutorTest(String jobName) { super(); this.jobName = jobName; } @Override public void run() { System.out.println("execute " + jobName); } public static void main(String[] args) { ScheduledExecutorService service = Executors.newScheduledThreadPool(10); long initialDelay1 = 1; long period1 = 1; // 从现在开始1秒钟之后,每隔1秒钟执行一次job1 service.scheduleAtFixedRate( new ScheduledExecutorTest("job1"), initialDelay1, period1, TimeUnit.SECONDS); long initialDelay2 = 1; long delay2 = 1; // 从现在开始2秒钟之后,每隔2秒钟执行一次job2 service.scheduleWithFixedDelay( new ScheduledExecutorTest("job2"), initialDelay2, delay2, TimeUnit.SECONDS); </span>}
<pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;"></span><pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;">}</span><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;">优缺:ScheduledExecutor:</span>是java 5 推出了基于线程池设计的 ScheduledExecutor。其设计思想是,每一个被调 度的任务都会由线程池中一个线程去执行,因此任务是并发执行的,相互之间不会受到干扰。需要注意的是,
只有当任务的执行时间到来时,ScheduedExecutor 才会真正启动一个线程,其余时间 ScheduledExecutor 都
是在轮询任务的状态。此定时器在抛出运行时异常时 虽然不会影响其他启动的任务但是本次的任务将会退出,
自此后不再执行。
三:基于框架Spring 的定时器:
如果你用了Spring 框架了,那么 本人极力推荐这种定时器,因为他集齐上述两种方式的有点于一身,并且没
有前两者的缺点。即:当前任务中抛出了运行时异常,当前任务线程不会退出。如果程序是每天定时任务,那
么在下一个任务运行的时间点任务仍然继续。
使用Spring 定时器需要的jar包:quartz-all-1.6.0.jar 和 jta.jar两个jar包。
配置文件:
在Spring 的业务逻辑bean容器配置文件中添加如下配置:
<!-- Spring 定时器 start -->
<bean id="activitytask" class="com.gop.service.activitytimer.ActivityTimer">
<property name="activityServiceImpl">
<ref bean="activityServiceImpl"/>
</property>
<property name="activityRecordServiceImpl">
<ref bean="activityRecordServiceImpl"/>
</property>
<property name="mailsServiceImpl">
<ref bean="mailsServiceImpl"/>
</property>
</bean>
//如果是一个普通的类,不需要注入dao层
<bean id=" activitytask " class=" com.gop.service.activitytimer.ActivityTimer "></bean>
//上面的两个bean节点根据自己的情况任选一个,下面的bean节点都是需要的。
<bean id="triggerB" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail">
<ref bean="jobDetailB"/>
</property>
<property name="cronExpression">
<value>0 34 17 * * ?</value>
</property>
</bean>
<bean id="jobDetailB" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetObject">
<ref bean="activitytask"/>
</property>
<property name="targetMethod" value="activityStart"/> //----①
</bean>
<bean id="scheduler" lazy-init="false" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers">
<list>
<ref bean="triggerB"/>
</list>
</property>
</bean>
实现类: com.gop.service.activitytimer.ActivityTimer
public class ActivityTimer {
public static final SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
public static final SimpleDateFormat sdfTim=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
private ActivityServiceImpl activityServiceImpl;
private MailsServiceImpl mailsServiceImpl;
private ActivityRecordServiceImpl activityRecordServiceImpl;
public ActivityTimer(){}
/**
* 定时器主处理方法
*/
public void activityStart() { //这里的方法名与配置文件中 ① 处名称一致
//每天晚24点 处理线上生效的活动.给玩家发奖。
// activityManage 为自定义处理逻辑方法。
activityManage();
}
}
时间的配置如下:
<value> 0 34 17 * * ?< value> 代表每天的17:34:00 开始执行任务
时间大小由小到大排列,从秒开始,顺序为 秒,分,时,天,月,年 *为任意 ?为无限制。由此上面所配置的内容就是,在每天的16点26分启动znrwdb方法
具体时间设定可参考
"0/10 * * * * ?" 每10秒触发
"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触发
每隔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 * * ?