Java定時執行任務

  在做項目時經常用遇到在指定的時間段執行某個操作,比如說項目中要求每天的00:00分計算當天的剩於庫存數或要求02:00開始清理某一個表中的數據,如何實現呢?下面我們就分幾步來完成這個任務。

 

1. 增加一個ServletContextListener監聽器,在服務器啓動時執行ServletContextListener。

 具體代碼如下:

   a. 實現ServletContextListener接口,實現接口中的方法。

public class TimerTaskListener implements ServletContextListener {
	private static final Logger logger = LogManager.getLogger(TimerTaskListener.class);
	@Override
	public void contextDestroyed(ServletContextEvent arg0) {
		logger.info("TimerTaskListener Destoryed!");
	}

	@Override
	public void contextInitialized(ServletContextEvent arg0) {
		// code written in here ...
	}
}

   b. 在web.xml中配置listener,這樣在服務器啓動時就可以調用TimerTaskListener監聽類了。

 

<listener> 
        <listener-class>com.jelly.commons.listener.TimerTaskListener</listener-class> 
</listener>

 2. 設計系統任務表,用於存放要執行的任務。

 表結構如下: 

TASK_ID FIRST_TIME PERIOD TASK_CLASS SERVICE_NAME
10001 2011-11-04 00:00:00 200000 com.jelly.service.task.TimerTaskServices1 任務1
10002 2011-11-03 02:00:00 500000 com.jelly.service.task.TimerTaskServices2 任務2

 

 TASK_ID(任務ID)

 FIRST_TIME(首次執行時間)

 PERIOD(執行頻率,單位毫秒,1天=24*60*60*1000)

 TASK_CLASS(任務的所在的類)

 SERVICE_NAME(任務名稱)

 

3. 新建一個TimerTask抽象類,該類主要是爲了讓用戶繼承使用(TASK_CLASS所指的父類),代碼如下:

 

public abstract class TimerTask {
	private static final Logger logger = LogManager.getLogger(TimerTaskListener.class);
	public abstract void execute(SystemTask task);
	public void start(final SystemTask task){
		// code written in here...
	}
}

 4. 編寫TimerTaskListener類中需要書寫的代碼。

 

public void contextInitialized(ServletContextEvent arg0) {
		new Thread(new Runnable() {
			@Override
			public void run() {
				DBHelper dbHelper = null;
				List<SystemTask> tasks = FastList.newInstance();
				try {
					Thread.sleep(30000);
					// query tasks
					dbHelper = DBHelper.getInstance();
					StringBuffer sql = new StringBuffer();
					sql.append("SELECT * FROM SYSTEM_TASK");
					tasks = dbHelper.select(sql.toString(), new BeanListHandler<SystemTask>(SystemTask.class));
					dbHelper.close();
				} catch (InterruptedException e) {
					logger.error(e.getMessage(), e);
				} catch (SQLException e) {
					logger.error(e.getMessage(), e);
				} finally {
					if(dbHelper != null){
						dbHelper.freeConnection();
					}
				}
				// execute task
				for (SystemTask task : tasks) {
					String className = task.getTaskClass();
					try {
						Class<?> clzz = Class.forName(className);
						TimerTask service = (TimerTask)clzz.newInstance();
						service.start(task);
					} catch (ClassNotFoundException e) {
						logger.error(e.getMessage(), e);
					} catch (InstantiationException e) {
						logger.error(e.getMessage(), e);
					} catch (IllegalAccessException e) {
						logger.error(e.getMessage(), e);
					} catch (Exception e) {
						logger.error(e.getMessage(), e);
					}
				}
			}
		}).start();
	}

 5. TimerTask類中的主要代碼。

 

public void start(final SystemTask task){
		final String taskId = task.getId();
		final long period = task.getPeriod();
		final Date firstTime = task.getFirstTime();
		Calendar c = Calendar.getInstance();
		c.setTime(firstTime);
		
		long dayMillisecond = c.getTimeInMillis();
		int hour = c.get(Calendar.HOUR_OF_DAY);
		int minute = c.get(Calendar.MINUTE);
		int second = c.get(Calendar.SECOND);
		int millisecond = c.get(Calendar.MILLISECOND);
		
		Calendar runtime = Calendar.getInstance();
		runtime.setTime(new Date());
		runtime.set(Calendar.HOUR_OF_DAY, hour);
		runtime.set(Calendar.MINUTE, minute);
		runtime.set(Calendar.SECOND, second);
		runtime.set(Calendar.MILLISECOND, millisecond);
		
		long nowMillisecond = runtime.getTimeInMillis();
		if(nowMillisecond >= dayMillisecond){
			if(nowMillisecond < new Date().getTime())
				runtime.add(Calendar.DAY_OF_MONTH, 1);
		}else{
			runtime.setTime(firstTime);
		}
		final String assignTime = UtilDateTime.toDateString(firstTime, UtilDateTime.FORMAT_DATE_TIME);
		final String executeTime = UtilDateTime.toDateString(runtime.getTime(), UtilDateTime.FORMAT_DATE_TIME);
		logger.info("task id (" + taskId + ") assign time : " + assignTime);
		logger.info("task id (" + taskId + ") execute time: " + executeTime);
		Timer timer = new Timer();
		timer.scheduleAtFixedRate(new java.util.TimerTask() {
			@Override
			public void run() {
				try {
					execute(task);
				} catch (Throwable e) {
					logger.error(e.getMessage(),e);
				}
			}
		}, runtime.getTime(), period);
	}

 

(未完)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章