面试突击:如何使用线程池执行定时任务?

开发 前端
scheduleWithFixedDelay 方法的使用和 scheduleAtFixedRate 类似,但执行效果完全不同,这个很容易理解如果效果一样就不用创建两个方法了。

在 Java 语言中,有两个线程池可以执行定时任务:ScheduledThreadPool 和 SingleThreadScheduledExecutor,其中 SingleThreadScheduledExecutor 可以看做是 ScheduledThreadPool 的单线程版本,它的用法和 ScheduledThreadPool 是一样的,所以本文重点来看 ScheduledThreadPool 线程池的使用。ScheduledThreadPool 执行定时任务的方法有以下 3 个:

  • 使用 schedule 方法执行定时任务,只执行一次定时任务。
  • 使用 scheduleAtFixedRate 方法执行定时任务,执行多次定时任务。
  • 使用 scheduleWithFixedDelay 方法执行定时任务,执行多次定时任务。

接下来我们看这 3 个方法的具体使用和区别。

1.schedule

schedule 方法只能执行一次定时任务,它需要传递 3 个参数:

  • 第 1 个参数:传递一个任务,Runnable 或 Callable 对象;
  • 第 2 个参数:添加定时任务后,再过多久开始执行定时任务;
  • 第 3 个参数:时间单位,配合参数 2 一起使用。

下面我们创建一个 3 秒以后执行的定时任务:

import java.time.LocalDateTime;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledThreadPoolExample {
public static void main(String[] args) throws InterruptedException {
// 创建 ScheduledThreadPool 线程池
ScheduledExecutorService threadPool = Executors.newScheduledThreadPool(10);
System.out.println("schedule 方法添加任务:" + LocalDateTime.now());
threadPool.schedule(new Runnable() {
@Override
public void run() {
System.out.println("执行 schedule 方法:" + LocalDateTime.now());
}
}, 3, TimeUnit.SECONDS); // 3s 之后执行

// 以下代码是给业务方法一个时间对照信息
TimeUnit.SECONDS.sleep(10); // 休眠 10s
System.out.println("当前时间:" + LocalDateTime.now());
}
}

以上程序的执行结果如下图所示:

从上述结果中可以看出,使用 schedule 方法只能执行一次定时任务。

2.scheduleAtFixedRate

scheduleAtFixedRate 方法可以执行多次定时任务,此方法需要 4 个参数:

  • 第 1 个参数:传递一个任务,Runnable 或 Callable 对象;
  • 第 2 个参数:添加定时任务后,再过多久开始执行定时任务;
  • 第 3 个参数:定时任务执行的时间间隔;
  • 第 4 个参数:时间单位,配合参数 2 和参数 3 一起使用。

下面我们创建一个 3 秒后执行的定时任务,每个定时任务执行的时间间隔为 2 秒,实现代码如下:

import java.time.LocalDateTime;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledThreadPoolExample {
public static void main(String[] args) throws InterruptedException {
// 创建 ScheduledThreadPool 线程池
ScheduledExecutorService threadPool = Executors.newScheduledThreadPool(10);
System.out.println("scheduleAtFixedRate 方法添加任务:" + LocalDateTime.now());
threadPool.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
System.out.println("执行 scheduleAtFixedRate 方法:" + LocalDateTime.now());
// 休眠 2s
try {
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},
3L, // 3s 后开始执行定时任务
2L, // 定时任务的执行间隔为 2s
TimeUnit.SECONDS); // 描述上面两个参数的时间单位
}
}

以上程序的执行结果如下图所示:

从上述结果可以看出,当任务添加成功之后,3s 后开始执行第一个定时任务,之后每隔 2s 执行一次定时任务。

3.scheduleWithFixedDelay

scheduleWithFixedDelay 方法的使用和 scheduleAtFixedRate 类似,但执行效果完全不同,这个很容易理解如果效果一样就不用创建两个方法了。scheduleWithFixedDelay 方法是在方法执行完成之后,再隔 N 秒执行下一个定时任务,和 scheduleAtFixedRate 的固定时间执行不同,scheduleWithFixedDelay 方法的执行受定时任务执行的时长影响,比如以下代码:

import java.time.LocalDateTime;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledThreadPoolExample {
public static void main(String[] args) throws InterruptedException {
// 创建 ScheduledThreadPool 线程池
ScheduledExecutorService threadPool = Executors.newScheduledThreadPool(10);
System.out.println("scheduleWithFixedDelay 方法添加任务:" + LocalDateTime.now());
threadPool.scheduleWithFixedDelay(new Runnable() {
@Override
public void run() {
System.out.println("执行 scheduleWithFixedDelay 方法:" + LocalDateTime.now());
// 休眠 2s
try {
TimeUnit.SECONDS.sleep(2);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},
3L, // 3s 后开始执行定时任务
2L, // 定时任务执行完 2s 之后,再执行下一个定时任务
TimeUnit.SECONDS); // 描述上面两个参数的时间单位
}
}

以上程序的执行结果如下图所示:

从上述结果可以看出,定时任务在 3s 之后开始执行,以后每隔 4s 执行一次,这 4s 包含了,定时任务执行花费的 2s,加上每隔 2s 执行一次的时间间隔,也就是说 scheduleWithFixedDelay 是在任务执行完 N 秒之后,再执行下一次定时任务。

总结

线程池执行定时任务的实现方法有 3 个:

使用 schedule 方法执行定时任务,只执行一次定时任务。

使用 scheduleAtFixedRate 方法执行定时任务,执行多次定时任务,它的执行时间间隔是固定的,不受定时任务执行时长影响(定时任务时间间隔 > 任务执行时间)。

使用 scheduleWithFixedDelay 方法执行定时任务,执行多次定时任务,它是在定时任务执行完之后,再隔 N 秒开始执行下一次定时任务,它的执行时间受定时任务执行时长影响。

责任编辑:武晓燕 来源: Java面试真题解析
相关推荐

2022-03-30 08:54:21

线程 Thread判断线程池任务Java

2024-02-28 09:54:07

线程池配置

2022-03-14 07:32:06

线程池拒绝策略自定义

2020-12-10 07:00:38

编程线程池定时任务

2022-03-07 07:33:16

线程池Java语言

2022-02-28 07:01:22

线程中断interrupt

2009-10-28 10:05:29

Ubuntucrontab定时任务

2022-03-23 08:51:21

线程池Java面试题

2024-06-20 13:59:26

2022-03-02 07:36:37

池化技术Java线程池

2024-09-09 15:09:30

2023-08-04 11:04:03

线程池项目开发

2024-11-04 16:01:01

2022-08-15 15:43:29

Linuxcron

2024-05-13 09:49:30

.NETQuartz库Cron表达式

2024-11-27 13:25:24

Rust线程池线程

2022-03-21 07:40:08

线程池Executors方式

2012-02-07 13:31:14

SpringJava

2023-12-08 14:42:17

Python开发

2010-03-10 15:47:58

crontab定时任务
点赞
收藏

51CTO技术栈公众号