最近做了下异步定时异常补偿任务,对定时任务用法做以下总结及比较,详细见注释。
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
/**
* @author zhipingxie
*/
public class Main {
public static void main(String[] args) {
// 定义间隔时间和延迟执行时间
final long timeInterval = 1000;
final long delay = 0;
/*
*第一种方式:使用普通线程
* 创建一个thread,执行的任务在while循环中,通过sleep方法来达到定时任务的效果
*优点:实现简单
* 缺点:sleep会一直占有对象锁,不便于控制
*/
Runnable runnable = new Runnable() {
@Override
public void run() {
while (true) {
// do something
System.out.println("do something in timer thread");
try {
Thread.sleep(timeInterval);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
Thread thread = new Thread(runnable);
thread.start();
System.out.println("do something in main");
/*
*第二种方式:使用java 自带的timer类
* 实现TimerTask接口,具体定时任务放到run方法中。使用Timer对象来调度timerTask(多个可选方法)
*优点:当启动和取消任务可以控制;第一次可以指定你想要的delay时间;线程安全
* 缺点:多线程并行处理定时任务时,Timer运行多个TimeTask时,只要其中之一没有捕获抛出的异常,其它任务便会自动终止运行。
* 使用ScheduledExecutorService则没有这个问题。 【使用ScheduledExecutorService代替Timer吧】
*/
// 创建一个TimerTask
TimerTask task = new TimerTask() {
@Override
public void run() {
// do something
System.out.println("do something in timerTask");
}
};
// 使用Timer实例调度任务
Timer timer = new Timer();
// 定义任务delay时间
timer.scheduleAtFixedRate(task,delay,timeInterval);
/*
*第三种方式:使用ScheduledExecutorService
* ScheduledExecutorService是从Java SE5的java.util.concurrent里,做为并发工具类被引进的,这是最理想的定时任务实现方式。
* 创建线程池,执行任务
*优点:相比于Timer的单线程,它是通过线程池的方式来执行任务的;可以灵活的设置delay时间;提供了良好的约定执行定时任务的时间间隔
*/
Runnable runnableTask = new Runnable() {
@Override
public void run() {
// do something
System.out.println("do something in task thread");
}
};
ScheduledExecutorService service = Executors.newSingleThreadScheduledExecutor();
service.scheduleAtFixedRate(runnableTask,delay,timeInterval,TimeUnit.MILLISECONDS);
}
}
顺带在这里说下,使用Timer实现定时任务有三个缺点:
- 上面提到的,由于Timer是单线程的,所以如果调度多个任务的情况是不适合的,如果其中一个任务抛出异常,其他任务也会被影响;
- Timer的时间是依赖系统时间的,如果系统时间变了,会出现一些执行上的变化。ScheduledExecutorService基于时间的延迟,不会由于系统时间的改变发生执行变化。
- 多个任务之间相互影响。在单线程中如果A任务所需时间超过了B任务delay时间,则B任务的执行会等A任务执行完之后才执行。