任务调度(二)——jdk自带的Timer 动态修改任务执行计划

       上一篇博文《任务调度(一)——jdk自带的Timer》中,简单介绍了一下Timer,本文将分享一下如何动态修改Timer制定的计划。

       先上代码:

package com.tgb.ccl.schema.dynamic;

import java.util.Date;

/**
 * 可动态修改的任务
 *
 * @author arron
 * @date 2015年5月9日 下午1:52:15
 * @version 1.0
 */
public class DynamicTimerTask extends  java.util.TimerTask {

	@Override
	public void run() {
		System.out.println("---------start--------");
		Date d = new Date();
		for(int i=0;i<2;i++){
			try {
				Thread.sleep(1000);
				System.out.println("已执行【"+(i+1)+"】秒钟,at: "+d.toLocaleString());
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		System.out.println("本次任务调度结束,at: "+new Date().toLocaleString());
		System.out.println("----------------------------------------------");
	}
}
package com.tgb.ccl.schema.dynamic;

import java.util.Calendar;
import java.util.Date;
import java.util.Timer;

/**
 * 任务调度管理器
 *
 * @author arron
 * @date 2015年5月9日 下午1:57:19
 * @version 1.0
 */
public class DynamicTaskManager {
	private static final long PERIOD = 5 * 1000;// 5秒钟

	/**
	 * 单例对象
	 */
	private static DynamicTaskManager taskManager = null;	

	/**
	 * 时间调度对象
	 */
	private static Timer timer = new Timer();

	/**
	 * 任务
	 */
	private static DynamicTimerTask task = null;

	static {
		taskManager = new DynamicTaskManager();
	}

	public static DynamicTaskManager getInstance(){
		if(taskManager==null){
			taskManager = new DynamicTaskManager();
		}
		return taskManager;
	}

	public DynamicTaskManager() {
	}

	@SuppressWarnings("deprecation")
	public void startTask(Date startTime, long period){

		System.out.println("设置启动时间: "+startTime.toLocaleString());
		//如果当前时间超过了设定时间,会立即执行一次
		task = new DynamicTimerTask();
		timer.schedule(task, startTime,period);

	}

	/**
	 * 启动定时器
	 */
	public void start() {
		//启动任务,10点40启动任务
		start(DateUtils.bookTime(10,40,0));
	}

	/**
	 * 启动定时器
	 */
	public void start(long preiod) {
		//启动任务,10点40启动任务
		start(DateUtils.bookTime(10,40,0),preiod);
	}

	/**
	 * 启动定时器
	 */
	public void start(Date startTime) {
		start(startTime,PERIOD);
	}

	/**
	 * 启动定时器
	 */
	public void start(Date startTime,long preiod) {
		startTask(startTime,preiod);
	}

	/**
	 * 重新启动
	 */
	public void restart() {
		clean();
		start();
	}

	/**
	 * 清空timer
	 */
	public void clean() {
		if(task != null){
			task.cancel();
		}
		timer.purge();
	}

	/**
	 * 停止任务
	 */
	public void stop(){
		System.out.println("--------任务正在停止---------");
		clean();
		System.out.println("---------任务已停止----------");
	}

	static class DateUtils{
		/**
		 * 增加或减少天数
		 *
		 * @param date
		 * @param CalendarFlag
		 * 				取值 Calendar.DAY_OF_MONTH, Calendar.HOUR_OF_DAY,
		 * 					   Calendar.MINUTE,Calendar.SECOND,Calendar.MILLISECOND
		 * @param num
		 * @return
		 */
		public static Date addDay(Date date, int CalendarFlag, int num) {
			Calendar startDT = Calendar.getInstance();
			startDT.setTime(date);
			startDT.add(CalendarFlag, num);
			return startDT.getTime();
		}

		/**
		 * 设定时间
		 *
		 * @param hour
		 * @param minute
		 * @param second
		 * @return
		 */
		public static Date bookTime(int hour, int minute, int second) {
			Calendar calendar = Calendar.getInstance();
			calendar.set(Calendar.HOUR_OF_DAY, hour);
			calendar.set(Calendar.MINUTE, minute);
			calendar.set(Calendar.SECOND, second);
			Date date = calendar.getTime();
			return date;
		}
	}

	@SuppressWarnings("deprecation")
	public static void main(String[] args) {
		DynamicTaskManager manager = DynamicTaskManager.getInstance();
		//启动任务,会立即执行一次,2s时执行完毕,5s时第二次执行,7s时第二次执行完毕
		manager.start();

		for(int i=0;i<8;i++){
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		//8s时,stop原任务,动态更改启动时间
		manager.stop();
		System.out.println("当前时间:"+new Date().toLocaleString());

		System.out.println("修改原计划,5s后重新执行");
		//5s后再启动,即13s时再启动
		manager.start(DateUtils.addDay(new Date(), Calendar.SECOND, 5));
	}
}

       运行结果如下:

       从结果中,我们可以看到,原先的计划是在14:46:40时开始,第二次是在14:46:45,每次执行2s中,也就是说如果不更改任务计划,那么任务将总会在0s或者5s的时候执行。但是我在main方法中,将原来的任务关闭,然后修改了启动时间,当前时间5s后启动新任务。当前时间为14:46:48,5s后执行了新计划。这就完成了任务的动态修改。

       首先说一下,如果关闭任务,stop方法完成了这个功能。要想任务不继续执行,必须将task的状态设置为cancel,然后调用timer的purge方法,将队列里的所有状态为cancel的task移除。这样就算到了执行时间,由于task已经移除,也就不会再执行了。如果使用了timer的cancel()方法,那么会将timer中所有的task全部移除掉。这点要注意一下。

       其实在项目中使用时,要比这个还要简单。直接修改startTask()方法,启动时间和间隔都是从数据库中取就ok了。只要想更改计划时,先配置好启动时间和间隔,然后自己写一个restart的方法,调用clean和startTask方法即可。

       有人问我Timer和Quartz框架的区别。那我就说一下我的理解。Timer毕竟是jdk自带的简易的任务调度工具类,跟Quartz比肯定是鸟枪与大炮的差距。Quartz的配置规则更加强大,更能满足我们的复杂需求,还允许多线程,这是Timer所比不了的。如果你就需要特别简单的任务调度,那么我觉得完全没有必要用Quartz。杀鸡焉用牛刀?!如果你的业务场景比较复杂,比如要求每个月的第4周的最后一个工作日要执行结算工资,如果最后一天是周六,就会向前提一天,在周五执行。这样的场景用Quartz要比Timer简单吧。

       不过具体用什么,还需要看项目。没有什么谁更牛,只有谁更适合。

时间: 2024-11-10 10:40:41

任务调度(二)——jdk自带的Timer 动态修改任务执行计划的相关文章

任务调度(一)——jdk自带的Timer

       说到任务调度,大家可能会想到Quartz框架,但是jdk自带的简单任务调度工具类,反而了解的人并不是很多.我觉得如果你的业务相对简单的话,没必要非得用Quartz等框架,使用Timer完全可以胜任的.简单来分享一下我了解的Timer. Timer是jdk中提供的一个定时器工具,使用的时候会在主线程之外起一个单独的线程执行指定的计划任务,可以指定执行一次或者反复执行多次. TimerTask是一个实现了Runnable接口的抽象类,代表一个可以被Timer执行的任务.        

在HTML中做好的页面放到aspx中,所有带js的动态的效果都变成静态的了??

问题描述 求解在HTML中做好的页面放到aspx中,所有带js的动态的效果都变成静态的了?? 解决方案 解决方案二:求解,此问题如何能看明白解决方案三:在HTML中做好的页面放到aspx中?你的aspx可能根本没正常动作.还有你是如何访问aspx页面的?是否已经发布到IIS?解决方案四:<%@PageLanguage="C#"AutoEventWireup="true"CodeFile="Default.aspx.cs"Inherits=&

使用JDK自带的WebService

WebService是个好东西,话不多说,干净利落 服务器端 来看下服务器端的结构: 先定义一个接口,用于暴露: ? 1 2 3 4 5 6 7 package com.abc.webservice; /**  * 对外暴露的接口.  */ public interface IWebService {     public String hello(String who); } 再定义这个接口的实现类: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18

java实现二维码([带]logo)的绘制和解析(swetake.util.Qrcode)

使用jar:qrcode.jar和源码twodimensioncode: 1:绘制二维码: package com.tsxs.tools.twodimensioncode; import java.awt.Color; import java.awt.Graphics2D; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; import java.io.OutputStrea

wsdl-java jdk自带的发布服务,wsimport命令生成客户端。

问题描述 java jdk自带的发布服务,wsimport命令生成客户端. 服务发布成功,能访问到服务也可以访问到wsdl文件,使用wsimport命令生成客户端代码的时候出错. 提示错误 [ERROR] Unexpected end of file from server Failed to read the WSDL document: http://localhost:8080/123/hell, because 1) cou ld not find the document; /2) t

JDK自带XML和java对象相互转换

下面使用的是JDK自带的类,没有引用任何第三方jar包.   Unmarshaller 类使客户端应用程序能够将 XML 数据转换为 Java 内容对象树. 备注:marshal(序列化.排列.整理) Marshaller 类使客户端应用程序能够将 Java 内容树转换回 XML 数据. package hb.jaxb;   import javax.xml.bind.annotation.XmlRootElement;   //1.需要转换的model对象一定要添加@XmlRootElemen

java方法块-android当中excute方法是jdk自带的吗

问题描述 android当中excute方法是jdk自带的吗 android当中excute方法是jdk自带的吗 这个方法是干什么用的 这个方法一定要抛出异常吗 解决方案 我自己去查查api文档了

Ext2.0本地模式动态修改combobox选择项

在很多时候,combobox的选择项已经下载到本地,只是存储方式不同,我们就需要动态修改combobox的选择项.例如有一颗树,树的节点就是combobox的选项,在Ext2.0中树的存储格式不是使用store的,而combobox必须使用store,这就需要进行数据转换.本文就以此作为例子,研究一下如何动态修改combobox的选择项. 我们先创建一个带几个选项的树: var root = new Ext.tree.TreeNode({ text: '选项', allowDrag:false,

微信的分享功能,每次调用分享功能的时候可以动态修改标题吗

问题描述 微信的分享功能,每次调用分享功能的时候可以动态修改标题吗 我现在做了一个h5的小游戏,每次用户玩完游戏后,用户要分享到朋友圈的时候,获取用户成绩,然后分享,是否配置好的标题修改呢 解决方案 可以 用js 修改 title 或者接入 jssdk Ajax.post({ url:"<%=basePath%>external/queryJssdk.action", dataType:"JSON", data:"url="+url,