在應用里經常都有用到在后台跑定時任務的需求。舉個例子,比如需要在服務后台跑一個定時任務來進行非實時計算,清除臨時數據、文件等。在本文里,我會給大家介紹3種不同的實現方法:
- 普通thread實現
- TimerTask實現
- ScheduledExecutorService實現
普通thread
這是最常見的,創建一個thread,然后讓它在while循環里一直運行着,通過sleep方法來達到定時任務的效果。這樣可以快速簡單的實現,代碼如下:
public class Task1 { public static void main(String[] args) { // run in a second final long timeInterval = 1000; Runnable runnable = new Runnable() { public void run() { while (true) { // ------- code for task to run System.out.println("Hello !!"); // ------- ends here try { Thread.sleep(timeInterval); } catch (InterruptedException e) { e.printStackTrace(); } } } }; Thread thread = new Thread(runnable); thread.start(); } }
用Timer和TimerTask
上面的實現是非常快速簡便的,但它也缺少一些功能。
用Timer和TimerTask的話與上述方法相比有如下好處:
- 當啟動和去取消任務時可以控制
- 第一次執行任務時可以指定你想要的delay時間
在實現時,Timer類可以調度任務,TimerTask則是通過在run()方法里實現具體任務。
Timer實例可以調度多任務,它是線程安全的。
當Timer的構造器被調用時,它創建了一個線程,這個線程可以用來調度任務:
import java.util.Timer; import java.util.TimerTask; public class Task2 { public static void main(String[] args) { TimerTask task = new TimerTask() { @Override public void run() { // task to run goes here System.out.println("Hello !!!"); } }; Timer timer = new Timer(); long delay = 0; long intevalPeriod = 1 * 1000; // schedules the task to be run in an interval timer.scheduleAtFixedRate(task, delay, intevalPeriod); } // end of main }
ScheduledExecutorService
ScheduledExecutorService是從Java SE 5的java.util.concurrent里,做為並發工具類被引進的,這是最理想的定時任務實現方式。
相比於上兩個方法,它有以下好處:
- 相比於Timer的單線程,它是通過線程池的方式來執行任務的
- 可以很靈活的去設定第一次執行任務delay時間
- 提供了良好的約定,以便設定執行的時間間隔
我們通過ScheduledExecutorService#scheduleAtFixedRate展示這個例子,通過代碼里參數的控制,首次執行加了delay時間:
import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.TimeUnit; public class Task3 { public static void main(String[] args) { Runnable runnable = new Runnable() { public void run() { // task to run goes here System.out.println("Hello !!"); } }; ScheduledExecutorService service = Executors .newSingleThreadScheduledExecutor(); service.scheduleAtFixedRate(runnable, 0, 1, TimeUnit.SECONDS); } }