1、创建一个servletContextListener类
package com.sinosoft.pis.listener;
/**
* 上下文监听器
*/
public class ContextListener2 implements ServletContextListener {
/**定时器服务*/
private BatchProService2 batchService = null;
public ContextListener2() {
System.out.println("监听器启动");
logger.warn("监听器启动 " + new Date().toString());
firstRun = true;
}
public void contextInitialized(ServletContextEvent event) {
try {
System.out.println("@@@### "+event.getServletContext().getServletContextName()+" init ###");
ExeSQL tExeSQL = new ExeSQL(); //一些操作
tExeSQL.execUpdateSQL("update batchmanage set ServiceFlag='0'");
batchService = new BatchProService2(); //继承timer类
batchService.submitData(tVData, tOperate)
} catch (Exception e) {
System.out.println("监听器初始化失败!");
logger.warn("监听器初始化失败");
e.printStackTrace();
}
}
public void contextDestroyed(ServletContextEvent event) {
try {
System.out.println("### "+event.getServletContext().getServletContextName()+" init ###");
stopTimerTask();
} catch (Exception e) {
System.out.println("监听器销毁失败!");
e.printStackTrace();
}
}
}
2、对timer定时类的继承。
package com.sinosoft.pis.batch;
/**
* 定时处理公用类
* 依据不同的batchcode,调用不同的批处理task
*/
public class BatchProService2 extends Timer{
/**
* 启动定时器
* @return
*/
public boolean submitData(){
output("定时器正在启动...");
try{
String taskClass = "com.sinosoft.pis.batch.CheckAccountYLTask,NULL,OrderInvalidLog"; //任务对象
String []keyValue = this.taskInfo.split(",");
taskClass = keyValue[0];
String flag = keyValue[1];
CheckAccountYLTask task = Class.forName(taskClass)。newInstance;
this.scheduleAtFixedRate(task , this.startTime, this.period); //不一定立即开始这个任务,到时间才可以执行。
output("定时器启动成功,正在运行批处理任务...");
return true;
}catch(Exception e){ }
}
}
3、执行TimerTask类
package com.sinosoft.pis.batch;
public class CheckAccountYLTask extends TimerTask {
public void run(){ //执行一系列任务
CheckTasklistener tCheckTasklistener=new CheckTasklistener();
if(!tCheckTasklistener.checktime(this.batchCode)){//時間未到,不執行
return;
}
ExeSQL eSQL = new ExeSQL(conn);
SSRS ssrs = null;
String tSQLQuery = "select ServiceFlag from BatchManage where BatchCode='" + batchCode + "' ";
ssrs = eSQL.execSQL(tSQLQuery);
if(ssrs.getMaxRow()==0){
mErrors.addOneError("并没有要启动的或者要关闭的定时器任务");
output("并没有要启动的或者要关闭的定时器任务");
insertDB(false,"并没有要启动的或者要关闭的定时器任务",null);
closeConnect(conn);
//this.cancel();//退出正在执行的任务
return;
}
}
import java.util.Timer;
import java.util.TimerTask;
import java.util.Date;
/**
* @author vincent
*/
public class TimerTest {
public static void main(String[] args) {
Timer t = new Timer();
//在5秒之后执行TimerTask的任务
t.schedule(new TimerTask(){
public void run()
{System.out.println("this is task you do1");}
},5*1000);
//在Date指定的特定时刻之后执行TimerTask的任务
SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2005/12/30 14:10:00");
t.schedule(new TimerTask(){
public void run()
{System.out.println("this is task you do2");}
},d1); //在2005/12/30 14:10:00后执行
//在Date指定的特定时刻之后,每隔1秒执行TimerTask的任务一次
Date d2 = new Date(System.currentTimeMillis()+1000);
t.schedule(new TimerTask(){
public void run()
{System.out.println("this is task you do3");}
},d2,1*1000);
//在3秒之后,每隔1秒执行TimerTask的任务一次
t.schedule(new TimerTask(){
public void run()
{System.out.println("this is task you do4");}
},3*1000,1*1000);
//在3秒之后,绝对每隔2秒执行TimerTask的任务一次
t.scheduleAtFixedRate(new TimerTask(){
public void run()
{System.out.println("this is task you do6");}
},3*1000,2*1000);
}
schedule和scheduleAtFixedRate的区别在于,如果指定开始执行的时间在当前系统运行时间之前,scheduleAtFixedRate会把已经过去的时间也作为周期执行,而schedule不会把过去的时间算上。
比如
SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2005/12/30 14:10:00");
t.scheduleAtFixedRate(new TimerTask(){
public void run()
{
System.out.println("this is task you do6");
}
},d1,3*60*1000);
间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,如果我在14:17:00分执行这个程序,那么会立刻打印3次
this is task you do6 //14:10
this is task you do6 //14:13
this is task you do6 //14:16
并且注意,下一次执行是在14:19 而不是 14:20。就是说是从指定的开始时间开始计时,而不是从执行时间开始计时。
但是上面如果用schedule方法,间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,那么在14:17:00分执行这个程序,则立即执行程序一次。并且下一次的执行时间是 14:20,而不是从14:10开始算的周期(14:19)。
}