一文搞懂,java中常用的定时任务框架-单体,从零开始系统化学Java

文章讲述了ScheduledExecutorService与Java内置Timer的区别,以及如何使用Spring的@Scheduled注解进行单线程串行和多线程并发任务调度,包括使用cron表达式和配置定时器线程池或异步线程池的解决方案。还提到了Quartz在定时任务调度中的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

import java.util.Date;

import java.util.concurrent.ScheduledExecutorService;

import java.util.concurrent.ScheduledThreadPoolExecutor;

import java.util.concurrent.TimeUnit;

/**

  • @Description: 2. ScheduledExecutorService代替Timer(jdk自带)

  • 多线程并行处理定时任务时,Timer运行多个TimeTask时,只要其中之一没有捕获抛出的异常,其它任务便会自动终止运行,

  • 使用ScheduledExecutorService则没有这个问题。

  • @Author: jianweil

  • @date: 2021/12/14 13:42

*/

public class ScheduledExecutorServiceTest {

public static void main(String[] args) throws InterruptedException {

//当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。

ScheduledExecutorService executorService = new ScheduledThreadPoolExecutor(1,

new BasicThreadFactory.Builder().namingPattern(“example-schedule-pool-%d”).daemon(false).build());

System.out.println(“begin:” + new Date());

// 参数:1、任务体 2、首次执行的延时时间 3、任务执行间隔 4、间隔时间单位

//延迟5秒执行,之后每3秒执行一次

executorService.scheduleAtFixedRate(new Runnable() {

@Override

public void run() {

//do something

System.out.println(“begin:” + new Date());

}

}, 5, 3, TimeUnit.SECONDS);

}

}

复制代码

四、Spring Task

=============

  • spring提供的类,可引入依赖:

org.springframework.boot

spring-boot-starter

复制代码

  • 开启定时任务:@EnableScheduling

  • 使用:在相应的任务方法前加上注解@Scheduled即可

4.1 单线程串行执行-@Scheduled


  • @Scheduled注解默认使同一个线程中串行执行,如果只有一个定时任务,这样做肯定没问题,当定时任务增多,如果一个任务卡死,会导致其他任务也无法执行。

  • 业务测试:

@Component

@EnableScheduling

public class SpringTaskTest {

@Scheduled(cron = “0/5 * * * * *”)

public void run() throws InterruptedException {

System.out.println(Thread.currentThread().getName() + “=====>>>>>使用cron {}” + (System.currentTimeMillis() / 1000));

}

}

复制代码

4.2 多线程并发运行-@Scheduled+配置定时器的程池(推荐)


  • 解决单线程串行执行任务的问题,需要配置定时器的程池,推荐这种方法

  • 配置并注入一个TaskScheduler类bean即可

  • 配置定时器的线程池类如下:

package com.ljw.springboottimer.springtask;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

import org.springframework.scheduling.TaskScheduler;

import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;

/**

  • @Description: 解决单线程串行执行 方式2:@Scheduled+配置定时器的线程池

  • @Author: jianweil

  • @date: 2021/12/14 14:44

*/

@Configuration

public class TaskSchedulerConfig {

/**

  • 初始化了一个线程池大小为 5 的 TaskScheduler, 避免了所有任务都用一个线程来执行

  • @return

*/

@Bean

public TaskScheduler taskScheduler() {

ThreadPoolTaskScheduler taskScheduler = new ThreadPoolTaskScheduler();

taskScheduler.setPoolSize(5);

taskScheduler.setThreadNamePrefix(“TaskSchedulerConfig-ljw”);

return taskScheduler;

}

}

复制代码

  • 业务测试

@Component

@EnableScheduling

public class SpringTaskTest {

@Scheduled(cron = “0/5 * * * * *”)

public void run() throws InterruptedException {

System.out.println(Thread.currentThread().getName() + “=====>>>>>使用cron {}” + (System.currentTimeMillis() / 1000));

}

@Scheduled(fixedRate = 5000)

public void run1() throws InterruptedException {

System.out.println(Thread.currentThread().getName() + “=====>>>>>使用fixedRate {}” + (System.currentTimeMillis() / 1000));

}

}

复制代码

4.3 多线程并发执行-@Scheduled+@Async+配置异步线程池


  • 解决单线程串行执行任务的问题,也可以结合异步注解@Async实现,但这种方法并不推荐,需要两个注解,代码编写的工作量大

  • 还可以解决fixedRate在遇到某些执行任务时间超过配置的时间隔,下次任务时间到了还要等待上次任务执行完成的情况,这是3.2不能解决的。

  • 配置异步线程池类如下:

package com.ljw.springboottimer.springtask;

import org.springframework.context.annotation.Configuration;

import org.springframework.scheduling.annotation.AsyncConfigurer;

import org.springframework.scheduling.annotation.EnableAsync;

import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import java.util.concurrent.Executor;

import java.util.concurrent.ThreadPoolExecutor;

/**

  • @Description: 解决单线程串行执行 方式1:@Scheduled+@Async+配置异步线程池

  • @Author: jianweil

  • @date: 2021/12/14 14:35

*/

@Configuration

@EnableAsync

public class AsyncConfig implements AsyncConfigurer {

/**

  • 定义@Async默认的线程池

  • ThreadPoolTaskExecutor不是完全被IOC容器管理的bean,可以在方法上加上@Bean注解交给容器管理,这样可以将taskExecutor.initialize()方法调用去掉,容器会自动调用

  • @return

*/

@Override

public Executor getAsyncExecutor() {

int processors = Runtime.getRuntime().availableProcessors();

//常用的执行器

ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();

//核心线程数

taskExecutor.setCorePoolSize(10);

taskExecutor.setMaxPoolSize(50);

//线程队列最大线程数,默认:50

taskExecutor.setQueueCapacity(100);

//线程名称前缀

taskExecutor.setThreadNamePrefix(“AsyncConfig-ljw-”);

taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());

//执行初始化(重要)

taskExecutor.initialize();

return taskExecutor;

}

}

复制代码

  • 业务测试需要加上@Async注解

@Component

@EnableScheduling

public class SpringTaskTest {

@Scheduled(cron = “0/5 * * * * *”)

@Async

public void run() throws InterruptedException {

System.out.println(Thread.currentThread().getName() + “=====>>>>>使用cron {}” + (System.currentTimeMillis() / 1000));

}

@Scheduled(fixedRate = 5000)

@Async

public void run1() throws InterruptedException {

System.out.println(Thread.currentThread().getName() + “=====>>>>>使用fixedRate {}” + (System.currentTimeMillis() / 1000));

}

}

复制代码

  • 如果同时配置了3.2配置定时器的程池和3.3配置异步线程池,并且注解使用了@Scheduled+@Async,则定时任务使用的线程池为:配置异步线程池

4.4 @Scheduled参数解析


  • cron:通过cron表达式来配置任务执行时间(默认是fixedDelay)

  • initialDelay :定义该任务延迟多少时间才开始第一次执行

  • fixedRate:定义一个按一定频率执行的定时任务。fixedRate 每次任务结束后会从任务编排表中找下一次该执行的任务,判断是否到时机执行,fixedRate的任务某次执行时间再长也不会造成两次任务实例同时执行,也要等到上次任务完成,判断是否到时机执行,到就立即执行,与线程池无关,除非用了@Async注解,使方法异步,即是使用5.3步骤的配置。(5.2是配置线程池,达不到效果)

  • fixedDelay:定义一个按一定频率执行的定时任务。fixedDelay总是在前一次任务完成后,延时固定时间长度然后再执行下一次任务

五、Quartz

========

在开发Quartz相关应用时,只要定义了Job(任务),JobDetail(任务描述),Trigger(触发器)和Scheduler(调度器),即可实现一个定时调度能力。

如果SpringBoot版本是2.0.0以后的,则在spring-boot-starter中已经包含了quart的依赖,则可以直接引入依赖:

org.springframework.boot

spring-boot-starter-quartz

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
img

最后我们该如何学习?

1、看视频进行系统学习

这几年的Crud经历,让我明白自己真的算是菜鸡中的战斗机,也正因为Crud,导致自己技术比较零散,也不够深入不够系统,所以重新进行学习是很有必要的。我差的是系统知识,差的结构框架和思路,所以通过视频来学习,效果更好,也更全面。关于视频学习,个人可以推荐去B站进行学习,B站上有很多学习视频,唯一的缺点就是免费的容易过时。

另外,我自己也珍藏了好几套视频资料躺在网盘里,有需要的我也可以分享给你:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

2、读源码,看实战笔记,学习大神思路

“编程语言是程序员的表达的方式,而架构是程序员对世界的认知”。所以,程序员要想快速认知并学习架构,读源码是必不可少的。阅读源码,是解决问题 + 理解事物,更重要的:看到源码背后的想法;程序员说:读万行源码,行万种实践。

Spring源码深度解析:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

Mybatis 3源码深度解析:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

Redis学习笔记:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

Spring Boot核心技术-笔记:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

3、面试前夕,刷题冲刺

面试的前一周时间内,就可以开始刷题冲刺了。请记住,刷题的时候,技术的优先,算法的看些基本的,比如排序等即可,而智力题,除非是校招,否则一般不怎么会问。

关于面试刷题,我个人也准备了一套系统的面试题,帮助你举一反三:

1年半经验,2本学历,Curd背景,竟给30K,我的美团Offer终于来了

只有技术过硬,在哪儿都不愁就业,“万般带不去,唯有业随身”学习本来就不是在课堂那几年说了算,而是在人生的旅途中不间断的事情。

人生短暂,别稀里糊涂的活一辈子,不要将就。

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img

链图片转存中…(img-JJHQU1UM-1712693982973)]

Spring Boot核心技术-笔记:

[外链图片转存中…(img-uXt6dABy-1712693982973)]

3、面试前夕,刷题冲刺

面试的前一周时间内,就可以开始刷题冲刺了。请记住,刷题的时候,技术的优先,算法的看些基本的,比如排序等即可,而智力题,除非是校招,否则一般不怎么会问。

关于面试刷题,我个人也准备了一套系统的面试题,帮助你举一反三:

[外链图片转存中…(img-tGG7rK9g-1712693982973)]

只有技术过硬,在哪儿都不愁就业,“万般带不去,唯有业随身”学习本来就不是在课堂那几年说了算,而是在人生的旅途中不间断的事情。

人生短暂,别稀里糊涂的活一辈子,不要将就。

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-qhFgeZu3-1712693982974)]

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值