在应用里经常都有用到在后台跑定时任务的需求。举个例子,比如需要在服务后台跑一个定时任务来进行垃圾回收(个人觉得用定时任务来跑垃圾回收不是很好的例子,比较常见的是用定时任务来进行非实时计算,清除临时数据、文件等)
在这里给大家介绍3种不同的实现方法:
- 普通thread实现
- TimerTask实现
- ScheduledExecutorService实现
普通thread
这是最常见的,创建一个thread,然后让它在while循环里一直运行着,通过sleep方法来达到定时任务的效果,代码如下:
1 /* 2 * Thread方式 3 * */ 4 Runnable runnable = new Runnable() { 5 6 @Override 7 public void run() { 8 while(true){ 9 try { 10 Thread.sleep(1000); 11 } catch (InterruptedException e) { 12 e.printStackTrace(); 13 } 14 System.out.println("hello,world..."); 15 } 16 } 17 }; 18 Thread t = new Thread(runnable); 19 t.start();
用Timer和TimerTask
上面的实现是非常快速简便的,但它也缺少一些功能。
用Timer和TimerTask的话与上述方法相比有如下好处:
①.当启动和去取消任务时可以控制
②.第一次执行任务时可以指定你想要的delay时间
在实现时,Timer类可以调度任务,TimerTask则是通过在run()方法里实现具体任务。
Timer实例可以调度多任务,它是线程安全的。
当Timer的构造器被调用时,它创建了一个线程,这个线程可以用来调度任务:
1 /** 2 * Timer和TimerTask 3 */ 4 TimerTask task = new TimerTask() { 5 6 @Override 7 public void run() { 8 System.out.println("hello,world"); 9 } 10 }; 11 Timer timer = new Timer(); 12 SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); 13 timer.schedule(task, format.parse("2014-04-09 21:19:00"), 1000);
ScheduledExecutorService
ScheduledExecutorService是从Java SE 5的java.util.concurrent里,做为并发工具类被引进的,这是最理想的定时任务实现方式。
相比于上两个方法,它有以下好处:
①.相比于Timer的单线程,它是通过线程池的方式来执行任务的
②.可以很灵活的去设定第一次执行任务delay时间
③.提供了良好的约定,以便设定执行的时间间隔
1 /** 2 * ScheduledExecutorService 3 */ 4 Runnable runnable = new Runnable() { 5 6 @Override 7 public void run() { 8 System.out.println("hello,world"); 9 } 10 }; 11 ScheduledExecutorService service = Executors.newSingleThreadScheduledExecutor(); 12 //service.schedule(runnable, 1, TimeUnit.SECONDS); 13 //如果给定秒,则delay,period都是以秒为单位 14 service.scheduleAtFixedRate(runnable, 5, 1, TimeUnit.SECONDS);