单点定时任务JDK原生自从JDK1.5之后 , 提供了ScheduledExecutorService代替TimerTask来执行定时任务 , 提供了不错的可靠性 。
public class SomeScheduledExecutorService {public static void main(String[] args) {// 创建任务队列 , 共 10 个线程ScheduledExecutorService scheduledExecutorService =Executors.newScheduledThreadPool(10);// 执行任务: 1秒 后开始执行 , 每 30秒 执行一次scheduledExecutorService.scheduleAtFixedRate(() -> {System.out.println("执行任务:" + new Date());}, 10, 30, TimeUnit.SECONDS);}}Spring TaskSpring Framework自带定时任务 , 提供了cron表达式来实现丰富定时任务配置 。新手推荐使用https://cron.qqe2.com/这个网站来匹配你的cron表达式 。
@Configuration@EnableSchedulingpublic class SomeJob {private static final Logger LOGGER = LoggerFactory.getLogger(SomeJob.class);/*** 每分钟执行一次(例:18:01:00 , 18:02:00)* 秒 分钟 小时 日 月 星期 年*/@Scheduled(cron = "0 0/1 * * * ? *")public void someTask() {//...}}单点的定时服务在目前微服务的大环境下 , 应用场景越来越局限 , 所以尝鲜一下分布式定时任务吧 。
基于 redis 实现相较于之前两种方式 , 这种基于Redis的实现可以通过多点来增加定时任务 , 多点消费 。但是要做好防范重复消费的准备 。
通过ZSet的方式将定时任务存放到ZSet集合中 , 并且将过期时间存储到ZSet的Score字段中 , 然后通过一个循环来判断当前时间内是否有需要执行的定时任务 , 如果有则进行执行 。
具体实现代码如下:
/** * Description: 基于Redis的ZSet的定时任务 .<br> * * @author mxy */@Configuration@EnableSchedulingpublic class RedisJob {public static final String JOB_KEY = "redis.job.task";private static final Logger LOGGER = LoggerFactory.getLogger(RedisJob.class);@Autowired private StringRedisTemplate stringRedisTemplate;/*** 添加任务.** @param task*/public void addTask(String task, Instant instant) {stringRedisTemplate.opsForZSet().add(JOB_KEY, task, instant.getEpochSecond());}/*** 定时任务队列消费* 每分钟消费一次(可以缩短间隔到1s)*/@Scheduled(cron = "0 0/1 * * * ? *")public void doDelayQueue() {long nowSecond = Instant.now().getEpochSecond();// 查询当前时间的所有任务Set<String> strings = stringRedisTemplate.opsForZSet().range(JOB_KEY, 0, nowSecond);for (String task : strings) {// 开始消费 taskLOGGER.info("执行任务:{}", task);}// 删除已经执行的任务stringRedisTemplate.opsForZSet().remove(JOB_KEY, 0, nowSecond);}}适用场景如下:
- 订单下单之后15分钟后 , 用户如果没有付钱 , 系统需要自动取消订单 。
- 红包24小时未被查收 , 需要延迟执退还业务;
- 某个活动指定在某个时间内生效&失效;
- 省去了MySQL的查询操作 , 而使用性能更高的Redis做为代替;
- 不会因为停机等原因 , 遗漏要执行的任务;
默认情况下Redis是不开启键空间通知的 , 需要我们通过config set notify-keyspace-events Ex的命令手动开启 。开启之后定时任务的代码如下:
自定义监听器【几种主流的分布式定时任务,你知道哪些?】
/*** 自定义监听器.*/public class KeyExpiredListener extends KeyExpirationEventMessageListener {public KeyExpiredListener(RedisMessageListenerContainer listenerContainer) {super(listenerContainer);}@Overridepublic void onMessage(Message message, byte[] pattern) {// channelString channel = new String(message.getChannel(), StandardCharsets.UTF_8);// 过期的keyString key = new String(message.getBody(), StandardCharsets.UTF_8);// todo 你的处理}}设置该监听器/** * Description: 通过订阅Redis的过期通知来实现定时任务 .<br> * * @author mxy */@Configurationpublic class RedisExJob {@Autowired private RedisConnectionFactory redisConnectionFactory;@Beanpublic RedisMessageListenerContainer redisMessageListenerContainer() {RedisMessageListenerContainer redisMessageListenerContainer = new RedisMessageListenerContainer();redisMessageListenerContainer.setConnectionFactory(redisConnectionFactory);return redisMessageListenerContainer;}@Beanpublic KeyExpiredListener keyExpiredListener() {return new KeyExpiredListener(this.redisMessageListenerContainer());}}
推荐阅读
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- 红茶与运动,红茶的主要功效与作用
- 游泳有哪些主要类别
- 发型|有这几种长相的女生,无论什么发型都好看!
- 杜伽K330w Plus评测:小巧精致、手感一流的热插拔机械键盘
- 虚拟主机和云服务器有什么区别?
- 心脏增大主动脉硬化的危害有哪些?
- 滑膜炎吃什么消炎药好
- AI知识点汇总
- 红茶放枣,盛茶的主要功效与作用是
- 高铁|直击郑州市民超市囤货:郑州高铁站多名工作人员感染 主城区实行足不出区
