1.应用场景:
在tomcat下,一个servlet启动时调用下面的start方法,关闭时调用stop方法
2.代码
import java.util.Calendar;
import java.util.Timer;
import java.util.TimerTask; private Timer timer; public ReportTask()
{
timer = new Timer();
} /**
* 定时检查数据库,发送邮件和短信报表
*/
public void run()
{
System.out.println("Start sending mail and sms reports");
...操作数据库,发送邮件和短信,关闭数据库资源
System.out.println("End sending mail and sms reports");
} /**
* 启动任务
*/
public void start()
{
Calendar now = Calendar.getInstance();
now.setTime(new Date());
Calendar start = Calendar.getInstance();
// 延时到下一个整点启动
start.set(Calendar.HOUR, now.get(Calendar.HOUR) + 1);
start.set(Calendar.MINUTE, 0);
start.set(Calendar.SECOND, 0);
start.set(Calendar.MILLISECOND, 0);
long delay = start.getTimeInMillis() - now.getTimeInMillis();
// 每个小时执行一次
timer.schedule(this, delay, 60 * 60 * 1000);
} /**
* 停止任务,可以由Servlet的Destroy方法触发
*/
public void stop()
{
timer.cancel();
}
}3.问题,上面的定时任务能正常起来,也能运行,但是运行大约2天左右后就不运行了。tomcat的程序其他功能都能正常,就是这个定时任务动不动就罢工。求教各位大侠如何定位和解决这个问题的思路。万分感谢。
在tomcat下,一个servlet启动时调用下面的start方法,关闭时调用stop方法
2.代码
import java.util.Calendar;
import java.util.Timer;
import java.util.TimerTask; private Timer timer; public ReportTask()
{
timer = new Timer();
} /**
* 定时检查数据库,发送邮件和短信报表
*/
public void run()
{
System.out.println("Start sending mail and sms reports");
...操作数据库,发送邮件和短信,关闭数据库资源
System.out.println("End sending mail and sms reports");
} /**
* 启动任务
*/
public void start()
{
Calendar now = Calendar.getInstance();
now.setTime(new Date());
Calendar start = Calendar.getInstance();
// 延时到下一个整点启动
start.set(Calendar.HOUR, now.get(Calendar.HOUR) + 1);
start.set(Calendar.MINUTE, 0);
start.set(Calendar.SECOND, 0);
start.set(Calendar.MILLISECOND, 0);
long delay = start.getTimeInMillis() - now.getTimeInMillis();
// 每个小时执行一次
timer.schedule(this, delay, 60 * 60 * 1000);
} /**
* 停止任务,可以由Servlet的Destroy方法触发
*/
public void stop()
{
timer.cancel();
}
}3.问题,上面的定时任务能正常起来,也能运行,但是运行大约2天左右后就不运行了。tomcat的程序其他功能都能正常,就是这个定时任务动不动就罢工。求教各位大侠如何定位和解决这个问题的思路。万分感谢。
public void contextInitialized(ServletContextEvent event) {
timer = new java.util.Timer(true);
event.getServletContext().log("定时器已启动");
timer.schedule(new MyTask(event.getServletContext()), 0, 60*60*1000);
event.getServletContext().log("已经添加任务调度表");
}
public void contextDestroyed(ServletContextEvent event) {
timer.cancel();
event.getServletContext().log("定时器销毁");
} 以上代码中, timer.schedule(new MyTask(event.getServletContext()), 0, 60*60*1000)这一行为定时器调度语句,其中MyTask是自定义需要被调度的执行任务(在我的财政数据中心项目中就是报表计算引擎入口),从java.util.TimerTask继承,下面会重点讲述,第三个参数表示每小时(即60*60*1000毫秒)被触发一次,中间参数0表示无延迟。其它代码相当简单,不再详细说明。 下面介绍MyTask的实现,上面的代码中看到了在构造MyTask时,传入了javax.servlet.ServletContext类型参数,是为记录Servlet日志方便而传入,因此需要重载MyTask的构造函数(其父类java.util.TimerTask原构造函数是没有参数的)。在timer.schedule()的调度中,设置了每小时调度一次,因此如果想实现调度任务每24小时被执行一次,还需要判断一下时钟点,以常量C_SCHEDULE_HOUR表示(晚上12点,也即0点)。同时为防止24小时执行下来,任务还未执行完(当然,一般任务是没有这么长的),避免第二次又被调度以引起执行冲突,设置了当前是否正在执行的状态标志isRunning。示例代码如下所示:private static final int C_SCHEDULE_HOUR = 0;
private static boolean isRunning = false;
private ServletContext context = null;
public MyTask(ServletContext context) {
this.context = context;
}
public void run() {
Calendar cal = Calendar.getInstance();
if (!isRunning) {
if (C_SCHEDULE_HOUR == cal.get(Calendar.HOUR_OF_DAY)) {
isRunning = true;
context.log("开始执行指定任务");//TODO 添加自定义的详细任务,以下只是示例
int i = 0;
while (i++ < 10) {
context.log("已完成任务的" + i + "/" + 10);
}
isRunning = false;
context.log("指定任务执行结束");
}
} else {
context.log("上一次任务执行还未结束");
}
}
上面代码中"//TODO……"之下四行是真正被调度执行的演示代码(在我的财政数据中心项目中就是报表计算过程),您可以换成自己希望执行的语句。 到这儿,ServletContextListener和MyTask的代码都已完整了。最后一步就是把ServletContextListener部署到您的Web工程中去,在您工程的web.xml配置文件中加入如下三行:
<listener>
<listener-class>com.test.ContextListener</listener-class>
</listener> 当然,上面的com.test得换成您自己的包名了。保存web.xml文件后,把工程打包部署到Tomcat中即可。任务会在每晚12点至凌晨1点之间被执行,上面的代码会在Tomcat的日志文件中记录如下:2003-12-05 0:21:39 开始执行指定任务
2003-12-05 0:21:39 已完成任务的1/10
……
2003-12-05 0:21:39 已完成任务的10/10
2003-12-05 0:21:39 指定任务执行结束以上援引自http://www.kupage.com/wpm/12/20040302/1701290000027kkaxl1y.htm
System.out.println("Start sending mail and sms reports");这个语句就没有执行,也就是run函数没有触发。另外,更正一下上面的启动和停止,也的确是如老兄所言的方法启动的。
public class MgrInit implements ServletContextListener
{
private ReportTask reportTask = null;
public void contextInitialized(ServletContextEvent sce)
{
reportTask = new ReportTask();
reportTask.start();
} public void contextInitialized(ServletContextEvent sce)
{
这个地方好像忘记stop了
}
}如果忘记stop了,重启tomcat,能出现我说的问题吗?希望从理论上说说,实际我会动手试验并把结果发上来的