其实定时执行任务Timer包也可以,但是Timer的缺点比较明显现在已经使用newScheduledThreadPool代替了;
Timer介绍:
在『任务调度线程池』功能加入之前,可以使用 java.util.Timer 来实现定时功能,Timer 的优点在于简单易用,但 由于所有任务都是由同一个线程来调度,因此所有任务都是串行执行的,同一时间只能有一个任务在执行,前一个 任务的延迟或异常都将会影响到之后的任务。
总的来说,对于一个Timer对象而言,如果你交给他多个任务的话,它还是始终使用一个线程来执行,并且可能出现线程异常终止或者是任务超时的情况;
Timer的缺点:
任务丢失:Timer对象始终使用一个线程来执行定时任务,也就意味着如果执行到一个任务的途中线程意外的终止了,那么后面的任务也就直接丢弃,不会执行了
任务超时:在安排的定时任务中,例如任务1需要在1点执行,任务2需要在1点03分执行,可是在执行任务1的时候却花费了半个小事,轮到任务2的时候,任务二已经不是在指定的时间内执行的了,这就是任务超时.
newScheduledThreadPool的优点:
自行定制线程数:newScheduledThreadPool可以自行决定你的线程数量,不需要担心你的线程数两不够
避免任务丢失:即使线程在执行任务的途中,线程出现了错误终止,线程也只是丢弃当前的任务,然后指向下一个任务,并不会像Timer一样直接线程卡死;
自由度高:综合程度上,newScheduledThreadPool确实实现的功能比Timer多的多;
简单代码演示:实现每一周的星期4执行指定的任务
package org.example.Pool;
import lombok.extern.slf4j.Slf4j;
import java.time.DayOfWeek;
import java.time.Duration;
import java.time.LocalDateTime;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
@Slf4j(topic = "c.TestSchedule")
public class TestSchedule {
//如何让每周四 18:00:00 定时执行任务
public static void main(String[] args) {
//获取当前时间
LocalDateTime now = LocalDateTime.now();
//获取本周周四时间
LocalDateTime time = now.withHour(18).withMinute(0).withSecond(0).withNano(0).with(DayOfWeek.THURSDAY);
//判断当前时间,如果当前时间已经过了本周周四了,那么把时间设定为下一周的周四
if (now.compareTo(time) > 0){
time = time.plusWeeks(1);//将时间加一周
}
//算出时间差 : 使用工具类Duration(专门做时间差的工具类)
long initailDelay = Duration.between(now, time).toMillis();
/**
* initailDelay :代表当前时间和周四的时间差
* period 一周的时间间隔
*/
long period = 1000*60*24*60*7;
ScheduledExecutorService pool = Executors.newScheduledThreadPool(1);
pool.scheduleAtFixedRate(()->{
log.debug("执行任务ing");
log.debug("执行任务完毕");
},initailDelay,period, TimeUnit.MILLISECONDS);
}
}