java 定时器

本文介绍如何使用Java定时器与Servlet侦听器实现定时任务,通过具体示例展示了如何创建定时器并将其与ServletContextListener接口结合,确保定时任务能够随应用启动和关闭。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

import java.io.IOException;

import java.util.Timer;

public class TimerTest {

public static void main(String[] args){

Timer timer = new Timer();

timer.schedule(new MyTask(), 1000, 2000);//在1秒后执行此任务,每次间隔2秒,如果传递一个Data参数,就可以在某个固定的时间执行这个任务.

while(true){//这个是用来停止此任务的,否则就一直循环执行此任务了

try {

int ch = System.in.read();

if(ch-'c'==0){

timer.cancel();//使用这个方法退出任务

}

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

static class MyTask extends java.util.TimerTask{

@Override

public void run() {

// TODO Auto-generated method stub

System.out.println("________");

}

}

}


  

    下面就Servlet侦听器结合Java定时器来讲述整个实现过程。要运用Servlet侦听器需要实现javax.servlet.ServletContextListener接口,同时实现它的contextInitialized(ServletContextEvent event)和contextDestroyed(ServletContextEvent event)两个接口函数。考虑定时器有个建立和销毁的过程,看了前面两个接口函数,就不容置疑的把建立的过程置入contextInitialized,把销毁的过程置入contextDestroyed了。

    我把ServletContextListener的实现类取名为ContextListener,在其内添加一个定时器,示例代码如下所示(为考虑篇幅,仅提供部分代码供读者参考):


    private java.util.Timer timer = null; 
    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("上一次任务执行还未结束"); 
        } 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值