//默认定时器名称
public Timer() {
this("Timer-" + serialNumber());
}
//以守护进程的方式运行此定时器
public Timer(boolean isDaemon) {
this("Timer-" + serialNumber(), isDaemon);
}
//自定义定时器名称
public Timer(String name) {
thread.setName(name);
thread.start();
}
public Timer(String name, boolean isDaemon) {
thread.setName(name);
thread.setDaemon(isDaemon);
thread.start();
}
schedule
//计划在指定的延迟后执行指定的任务,以毫秒为单位
public void schedule(TimerTask task, long delay) {
if (delay < 0)
throw new IllegalArgumentException("Negative delay.");
sched(task, System.currentTimeMillis()+delay, 0);
}
//计划在指定时间执行指定任务。如果时间已过,则计划立即执行任务
// long delay 跟 Date time 的区别:delay 必须大于0,即未来时间;time可以是过去时间
public void schedule(TimerTask task, Date time) {
sched(task, time.getTime(), 0);
}
//将指定的任务安排为重复的固定延迟执行,从指定的延迟之后开始。
//随后大约每隔一定的时间间隔重复执行。都是以毫秒为单位
public void schedule(TimerTask task, long delay, long period) {
if (delay < 0)
throw new IllegalArgumentException("Negative delay.");
if (period <= 0)
throw new IllegalArgumentException("Non-positive period.");
sched(task, System.currentTimeMillis()+delay, -period);
}
public void schedule(TimerTask task, Date firstTime, long period) {
if (period <= 0)
throw new IllegalArgumentException("Non-positive period.");
sched(task, firstTime.getTime(), -period);
}
//以毫秒为单位,在指定的时间和指定的时间段安排指定的计时器任务执行。
//如果周期为正,则计划重复执行任务;如果周期为零,则计划一次性执行任务
private void sched(TimerTask task, long time, long period) {
if (time < 0)
throw new IllegalArgumentException("Illegal execution time.");
// Constrain value of period sufficiently to prevent numeric
// overflow while still being effectively infinitely large.
if (Math.abs(period) > (Long.MAX_VALUE >> 1))
period >>= 1;
synchronized(queue) {
if (!thread.newTasksMayBeScheduled)
throw new IllegalStateException("Timer already cancelled.");
synchronized(task.lock) {
if (task.state != TimerTask.VIRGIN)
throw new IllegalStateException(
"Task already scheduled or cancelled");
task.nextExecutionTime = time;
task.period = period;
task.state = TimerTask.SCHEDULED;
}
queue.add(task);
if (queue.getMin() == task)
queue.notify();
}
}
public static void main(String[] args) throws InterruptedException {
demo01();
}
public static void demo01() throws InterruptedException {
Timer timer = new Timer();
log.info("=============第一次启动定时器=============");
timer.schedule(new TimerTask() {
@Override
public void run() {
log.info("一次性执行,启动定时器延迟3秒执行");
}
}, 3000);
Thread.sleep(5000);
log.info("=============第二次启动定时器=============");
timer.schedule(new TimerTask() {
@Override
public void run() {
try {
Thread.sleep(3000);
log.info("周期性执行,启动定时器延迟3秒执行,之后每隔2秒执行一次,但如果任务时间超过2秒则以任务时间为间隔");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, 3000, 2000);
}
scheduleAtFixedRate
//将指定的任务安排为重复的固定速率执行,从指定的延迟后开始。
//随后大约每隔一定的时间间隔进行,并按规定的时间间隔进行。
public void scheduleAtFixedRate(TimerTask task, long delay, long period) {
if (delay < 0)
throw new IllegalArgumentException("Negative delay.");
if (period <= 0)
throw new IllegalArgumentException("Non-positive period.");
sched(task, System.currentTimeMillis()+delay, period);
}
public void scheduleAtFixedRate(TimerTask task, Date firstTime,
long period) {
if (period <= 0)
throw new IllegalArgumentException("Non-positive period.");
sched(task, firstTime.getTime(), period);
}
public static void demo02() {
Timer timer = new Timer();
log.info("=============开始启动定时器=============");
timer.scheduleAtFixedRate(new TimerTask() {
@Override
public void run() {
try {
Thread.sleep(3000);
log.info("周期性执行,启动定时器延迟3秒执行,之后每隔2秒执行一次,但如果任务时间超过2秒则以任务时间为间隔");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}, 3000, 2000);
}
2、ScheduledThreadPoolExecutor
@Bean(name = "scheduledPoolTaskExecutor", destroyMethod = "shutdown")
public ScheduledThreadPoolExecutor scheduledThreadPoolExecutor() {
// 第一个参数:核心线程数
// 第二个参数:线程工厂(此处用来设置线程名)
// 第三次参数:饱和策略
// 饱和策略
// AbortPolicy:直接抛出java.util.concurrent.RejectedExecutionException异常
// CallerRunsPolicy:若已达到待处理队列长度,将由主线程直接处理请求
// DiscardOldestPolicy:抛弃旧的任务;会导致被丢弃的任务无法再次被执行
// DiscardPolicy:抛弃当前任务;会导致被丢弃的任务无法再次被执行
return new ScheduledThreadPoolExecutor(10,
new ThreadFactory() {
int i = 0;
@Override
public Thread newThread(Runnable r) {
return new Thread(r, "scheduled-" + i++);
}
},
new ThreadPoolExecutor.DiscardPolicy()
);
}
@Slf4j
@Component
public class PoolTaskDemo {
@Resource(name = "scheduledPoolTaskExecutor")
private ScheduledThreadPoolExecutor scheduled;
@PostConstruct
public void scheduledPoolTaskExecutor() {
schedule();
// scheduleAtFixedRate();
// scheduleWithFixedDelay();
}
public void schedule() {
log.info("调用前");
scheduled.schedule(new Runnable() {
@Override
public void run() {
log.info("方法调用,延迟5秒执行一次, 只执行一次");
}
}, 5000, TimeUnit.MILLISECONDS);
}
}
public void scheduleAtFixedRate() {
log.info("调用前");
scheduled.scheduleAtFixedRate(new Runnable() {
@SneakyThrows
@Override
public void run() {
log.info("方法调用,延迟5秒首次执行,之后每过3秒轮询, 不受run方法执行耗时影响");
Thread.sleep(2000);
}
}, 5000, 3000, TimeUnit.MILLISECONDS);
}
public void scheduleWithFixedDelay() {
log.info("调用前");
scheduled.scheduleWithFixedDelay(new Runnable() {
@SneakyThrows
@Override
public void run() {
log.info("方法调用,延迟5秒首次执行,之后每过3秒轮询,受run方法执行耗时影响");
Thread.sleep(2000);
}
}, 5000, 3000, TimeUnit.MILLISECONDS);
}
3、@Scheduled
@Configuration
@EnableScheduling // 使@Scheduled生效
public class ScheduledConfig implements SchedulingConfigurer {
@Override
public void configureTasks(ScheduledTaskRegistrar scheduledTaskRegistrar) {
scheduledTaskRegistrar.setScheduler(scheduledThreadPoolExecutor());
}
@Bean(name = "scheduledPoolTaskExecutor", destroyMethod = "shutdown")
public ScheduledThreadPoolExecutor scheduledThreadPoolExecutor() {
// 第一个参数:核心线程数
// 第二个参数:线程工厂(此处用来设置线程名)
// 第三次参数:饱和策略
// 饱和策略
// AbortPolicy:直接抛出java.util.concurrent.RejectedExecutionException异常
// CallerRunsPolicy:若已达到待处理队列长度,将由主线程直接处理请求
// DiscardOldestPolicy:抛弃旧的任务;会导致被丢弃的任务无法再次被执行
// DiscardPolicy:抛弃当前任务;会导致被丢弃的任务无法再次被执行
return new ScheduledThreadPoolExecutor(10,
new ThreadFactory() {
int i = 0;
@Override
public Thread newThread(Runnable r) {
return new Thread(r, "scheduled-" + i++);
}
},
new ThreadPoolExecutor.DiscardPolicy()
);
}
}
fixedRate
@Slf4j
@Component
public class ScheduledDemo {
@Scheduled(fixedRate = 5000)
public void fixedRate() throws InterruptedException {
log.info("项目启动时立即执行第一次,之后每过5秒固定执行一次,不受方法执行耗时影响");
Thread.sleep(2000);
}
}
fixedDelay
@Scheduled(fixedDelay = 5000)
public void fixedDelay() throws InterruptedException {
log.info("项目启动时立即执行第一次,方法内部执行完成之后过5秒再次执行,受方法执行耗时影响");
Thread.sleep(2000);
}
initialDelay
@Scheduled(initialDelay = 5000,fixedRate = 10000)
public void initialDelayFixedRate() throws InterruptedException {
log.info("项目启动时延迟5秒才执行第一次,在第一次执行后每过10秒固定执行一次,不受方法执行耗时影响");
Thread.sleep(2000);
}
cron
@Scheduled(cron = "0/5 * * * * ? ")
public void cron() throws InterruptedException {
log.info("项目启动时延迟5秒才执行第一次,在第一次执行时每过5秒固定执行一次,不受方法执行耗时影响");
Thread.sleep(2000);
}
@Scheduled(cron = "5 * * * * ? ")
public void cron2() throws InterruptedException {
log.info("每分钟的刚好第5秒时,执行一次,不受方法执行耗时影响");
Thread.sleep(2000);
}
cron表达式
cronExpression定义时间规则,Cron表达式由6或7个空格分隔的时间字段组成:秒 分钟 小时 日期 月份 星期 年(可选)
字段 允许值 允许的特殊字符
秒 0-59 , - * /
分 0-59 , - * /
小时 0-23 , - * /
日期 1-31 , - * ? / L W C
月份 1-12 , - * /
星期 1-7 , - * ? / L C #
年 1970-2099 , - * /
特殊符号 代表含义
, 枚举
- 区间
* 任意
/ 步长
? 日/星期冲突匹配
L 最后
W 工作日
C 和Calendar联系后计算过的值
# 星期 4#2 第2个星期四
如:
"*": 字符被用来指定所有的值。如:在分钟字段域里表示“每分钟”
"?":字符只在日期域和星期域中使用。它被用来指定“非明确的值”。
"-":字符被用来指定一个范围。如:“10-12”在小时域意味着“10点、11点、12点”。
",":字符被用来指定另外的值。如:“MON,WED,FRI”在星期域里表示”星期一、星期三、星期五”。
"/":字符用于指定增量。如:"0/5"在秒域0秒开始,每过5秒触发一次,"5/15"在分钟域表示每小时的5,20,35和50。
符号"*"在"/"前面(如:*/10)等价于0在“/”前面(如:0/10)。
L是‘last’的省略写法可以表示day-of-month和day-of-week域,但在两个字段中的意思不同,例如day-of- month域中表示一个月的最后一天。
如果在day-of-week域表示‘7’或者‘SAT’,如果在day-of-week域中前面加上数字,它表示 一个月的最后几天,例如‘6L’就表示一个月的最后一个星期五。
字符“W”只允许日期域出现。这个字符用于指定日期的最近工作日。例如:如果你在日期域中写 “15W”,表示:这个月15号最近的工作日。
所以,如果15号是周六,则任务会在14号触发。如果15好是周日,则任务会在周一也就是16号触发。
如果 是在日期域填写“1W”即使1号是周六,那么任务也只会在下周一,也就是3号触发,“W”字符指定的最近工作日是不能够跨月份的。
字符“W”只能配合一个 单独的数值使用,不能够是一个数字段,如:1-15W是错误的。
“L”和“W”可以在日期域中联合使用,LW表示这个月最后一周的工作日。
字符“#”只允许在星期域中出现。这个字符用于指定本月的某某天。例如:“6#3”表示本月第三周的星期五(6表示星期五,3表示第三周)。
“2#1”表示本月第一周的星期一。“4#5”表示第五周的星期三。
字符“C”允许在日期域和星期域出现。这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历” 关联,则等价于所有包含的“日历”。
如:日期域是“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。星期域是“1C”表示关联“日历” 中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。
经典例子:
"0 0 10,14,16 * * ?" 每天上午10点,下午2点,4点
"0 0/30 9-17 * * ?" 朝九晚五工作时间内每半小时
"0 0 12 ? * WED" 表示每个星期三中午12点
"0 0 12 * * ?" 每天中午12点触发
"0 15 10 ? * *" 每天上午10:15触发
"0 15 10 * * ?" 每天上午10:15触发
"0 15 10 * * ? *" 每天上午10:15触发
"0 15 10 * * ? 2005" 2005年的每天上午10:15触发
"0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟触发
"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发
"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发
"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发
"0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发
"0 15 10 15 * ?" 每月15日上午10:15触发
"0 15 10 L * ?" 每月最后一日的上午10:15触发
"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发
"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发
"30 * * * * ?" 每半分钟触发任务
"30 10 * * * ?" 每小时的10分30秒触发任务
"30 10 1 * * ?" 每天1点10分30秒触发任务
"30 10 1 20 * ?" 每月20号1点10分30秒触发任务
"30 10 1 20 10 ? *" 每年10月20号1点10分30秒触发任务
"30 10 1 20 10 ? 2011" 2011年10月20号1点10分30秒触发任务
"30 10 1 ? 10 * 2011" 2011年10月每天1点10分30秒触发任务
"30 10 1 ? 10 SUN 2011" 2011年10月每周日1点10分30秒触发任务
"15,30,45 * * * * ?" 15秒,30秒,45秒时触发任务
"15-45 * * * * ?" 15到45秒内,每秒都触发任务
"15/5 * * * * ?" 每分钟的每15秒开始触发,每隔5秒触发一次
"15-30/5 * * * * ?" 每分钟的15秒到30秒之间开始触发,每隔5秒触发一次
"0 0/3 * * * ?" 每小时的第0分0秒开始,每三分钟触发一次
"0 15 10 ? * MON-FRI" 星期一到星期五的10点15分0秒触发任务
"0 15 10 L * ?" 每个月最后一天的10点15分0秒触发任务
"0 15 10 LW * ?" 每个月最后一个工作日的10点15分0秒触发任务
"0 15 10 ? * 5L" 每个月最后一个星期四的10点15分0秒触发任务
"0 15 10 ? * 5#3" 每个月第三周的星期四的10点15分0秒触发任务
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)