复制代码
-
开启定时任务:@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
复制代码
5.1. 创建任务类
-
方式1:实现Job类的execute方法即可实现一个任务(推荐)
-
任务1如下:
package com.ljw.springboottimer.quartz.do1;
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import java.util.Date;
/**
-
@Description: 我的定时任务-方法1
-
@Author: jianweil
-
@date: 2021/12/14 16:06
*/
public class MyTaskService1 implements Job {
@Override
public void execute(JobExecutionContext jobExecutionContext) throws JobExecutionException {
System.out.println(Thread.currentThread().getName() + “------ Job ------” + new Date());
}
}
复制代码
-
方式2:继承QuartzJobBean类重写方法即可实现一个任务
-
任务2如下:
package com.ljw.springboottimer.quartz.do1;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
import org.springframework.scheduling.quartz.QuartzJobBean;
import java.util.Date;
/**
-
@Description: 我的定时任务-方法2
-
@Author: jianweil
-
@date: 2021/12/14 16:06
*/
public class MyTaskService2 extends QuartzJobBean {
@Override
protected void executeInternal(JobExecutionContext context) throws JobExecutionException {
System.out.println(Thread.currentThread().getName() + “—QuartzJobBean-----” + new Date());
}
}
复制代码
5.2. 配置任务描述和触发器
-
配置类要分别要为每个任务声明两个bean
-
1.JobDetail(任务描述)
-
2.Trigger(触发器)
-
配置调度器信息使用SimpleScheduleBuilder或者CronScheduleBuilder
package com.ljw.springboottimer.quartz.do1;
import org.quartz.*;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.Date;
/**
-
@Description: 每个任务都要两步配置
-
1.配置任务描述JobDetail 2. 配置触发器Trigger
-
@Author: jianweil
-
@date: 2021/12/14 16:08
*/
@Configuration
public class QuartzConfig {
/**
-
创建任务1的 JobDetail1
-
@return
*/
@Bean
public JobDetail teatQuartzDetail1() {
return JobBuilder.newJob(MyTaskService1.class)
//job的描述
.withDescription(“this is a job1”)
//job 的name和group
.withIdentity(“myTrigger1”, “myTriggerGroup1”)
.storeDurably().build();
}
/**
-
创建任务2的 JobDetail2
-
@return
*/
@Bean
public JobDetail teatQuartzDetail2() {
return JobBuilder.newJob(MyTaskService2.class)
//job的描述
.withDescription(“this is a job2”)
//job 的name和group
.withIdentity(“myTrigger2”, “myTriggerGroup2”)
.storeDurably().build();
}
/**
-
创建任务1的 Trigger1
-
@return
*/
@Bean
public Trigger testQuartzTrigger1() {
//使用SimpleScheduleBuilder或者CronScheduleBuilder
SimpleScheduleBuilder scheduleBuilder = SimpleScheduleBuilder.simpleSchedule()
//设置时间周期单位秒
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
总结
在这里,由于面试中MySQL问的比较多,因此也就在此以MySQL为例为大家总结分享。但是你要学习的往往不止这一点,还有一些主流框架的使用,Spring源码的学习,Mybatis源码的学习等等都是需要掌握的,我也把这些知识点都整理起来了
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)
总结
在这里,由于面试中MySQL问的比较多,因此也就在此以MySQL为例为大家总结分享。但是你要学习的往往不止这一点,还有一些主流框架的使用,Spring源码的学习,Mybatis源码的学习等等都是需要掌握的,我也把这些知识点都整理起来了
[外链图片转存中…(img-RKhiFnwj-1713434664006)]
[外链图片转存中…(img-9EFGbYjy-1713434664006)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!