在
boot
环境中,一般来说,要实现定时任务,我们有两中方案,一种是使用
Spring
自带的定时
任务处理器
@Scheduled
注解,另一种就是使用第三方框架
Quartz
,
Spring Boot
源自
Spring+SpringMVC
,因此天然具备这两个
Spring
中的定时任务实现策略,当然也支持
Quartz
(不做过多介绍)。
1.定时任务 @Scheduled
@Configuration //1.主要用于标记配置类
@EnableScheduling // 2.开启定时任务
public class SaticScheduleTask {
@Resource
UserService userService;
//3.添加定时任务 每隔5秒调用一次
@Scheduled(cron = "0/5 * * * * ?")
public void configureTasks() {
userService.addUser();
System.err.println("执行静态定时任务时间: " + LocalDateTime.now());
}
}
cron = [
秒
] [
分
] [
小时
] [
日
] [
月
] [
周
] [
年
]
月份中的日期和星期可能会起冲突,因此在配置时这两个得有一个是
?
?
:表示不指定值,即不关心某个字段的取值时使用。需要注意的是,月份中的日期和星期可能会
起冲突,因此在配置时这两个得有一个是?
*
:表示所有值,例如
:
在秒的字段上设置
*,
表示每一秒都会触发
,
:用来分开多个值,例如在周字段上设置
"MON,WED,FRI"
表示周一,周三和周五触发
-
:表示区间,例如在秒上设置
"10-12",
表示
10,11,12
秒都会触发
/
:用于递增触发,如在秒上面设置
"5/15"
表示从
5
秒开始,每增
15
秒触发
(5,20,35,50)
#
:序号
(
表示每月的第几个周几
)
,例如在周字段上设置
"6#3"
表示在每月的第三个周六,
(
用 在母
亲节和父亲节再合适不过了
)
周字段的设置,若使用英文字母是不区分大小写的 ,即
MON
与
mon
相同
L
:表示最后的意思。在日字段设置上,表示当月的最后一天
(
依据当前月份,如果是二月还会自
动判断是否是润年
),
在周字段上表示星期六,相当于
"7"
或
"SAT"
(注意周日算是第一天)。如果
在
"L"
前加上数字,则表示该数据的最后一个。例如在周字段上设置
"6L"
这样的格式
,
则表示
"
本月
最后一个星期五
"
W
:表示离指定日期的最近工作日
(
周一至周五
)
,例如在日字段上设置
"15W"
,表示离每月
15
号最
近的那个工作日触发。如果
15
号正好是周六,则找最近的周五
(14
号
)
触发
,
如果
15
号是周未,则找
最近的下周一
(16
号
)
触发,如果
15
号正好在工作日
(
周一至周五
)
,则就在该天触发。如果指定格式
为
"1W",
它则表示每月
1
号往后最近的工作日触发。如果
1
号正是周六,则将在
3
号下周一触发。
(
注,
"W"
前只能设置具体的数字
,
不允许区间
"-")
L
和
W
可以一组合使用。如果在日字段上设置
"LW",
则表示在本月的最后一个工作日触发
0 0 2 1 ?
:表示在每月
1
日的凌晨
2
点执行
0 15 10 ? * MON-FRI
:表示周一到周五每天上午
10:15
执行
0 15 10 ? 6L 2019-2020
:表示
2019-2020
年的每个月的最后一个星期五上午
10:15
执行
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 ? 2019
:
2019
年的每天上午
10:15
执行
创建多线程定时任务
@Configuration //1.主要用于标记配置类,兼备Component的效果。
@EnableScheduling // 2.开启定时任务
@EnableAsync // 2.开启多线程
public class SaticScheduleTask {
@Resource
UserService userService;
//3.添加定时任务
@Scheduled(cron = "0/5 * * * * ?")
@Async //异步方法 异步调用 默认为同步
@Transactional //添加事务
public void configureTasks() {
userService.addUser();
System.err.println("执行静态定时任务时间: " + LocalDateTime.now());
}
}
根据
Spring
的文档说明,默认采用的是单线程的模式的。所以在
Java
应用中,绝大多数情况下都
是通过同步的方式来实现交互处理的。
那么当多个任务的执行势必会相互影响。例如,如果
A
任务执行时间比较长,那么
B
任务必须等到
A
任务执行完毕后才会启动执行。又如在处理与第三方系统交互的时候,容易造成响应迟缓的情
况,之前大部分都是使用多线程来完成此类任务,其实,在
spring3.x
之后,已经内置了
@Async
来完美解决这个问题。
2.拦截器
SpringMvc
的处理拦截器类似于
Servlet
开发中的过滤器
Filter
,用于对处理器进行预处理和后处理,开
发可以自己定义一些拦截器来实现特定功能。
###
拦截器与过滤器的区别
过滤器
:
servlet
规范中的一部分,任何
java web
程序都可以使用。
在
url-pattern
中配置之后,可以对所要访问的资源进行拦截。
拦截器
:
拦截器在
SpringMvc
框架自己的,只有使用了
SpringMvc
框架工程才能使用。
拦截器只会拦截访问控制器的方法,如果访问的是
js
,
css
,
image...
是不会进行拦截的。
##SpringBoot
中使用拦截器
1. 实现HandlerInterceptor接口
@Component
public class LoginInterceptor implements HandlerInterceptor {
//preHandle是请求执行前执行的
@Override
public boolean preHandle(HttpServletRequest httpServletRequest,
HttpServletResponse httpServletResponse, Object o) throws Exception {
System.out.println("进入拦截器");
return true;
}
//postHandler是请求结束执行的 当preHandle返回true才会执行
public void postHandle.....
//afterCompletion是视图渲染完成后才执行
public void afterCompletion.....
}
2.实现WebMvcConfigurer接口配置拦截路径
三种方式:
1. 继承WebMvcConfigurerAdapter spring5.0 以弃用,不推荐
2. 实现WebMvcConfigurer 推荐
3. 继承WebMvcConfigurationSupport 会导致springboot自动配置失效
@Configuration
public class WebJavaBeanConfiguration implements WebMvcConfigurer {
@Autowired
private LoginInterceptor loginInterceptor;
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(loginInterceptor)
.addPathPatterns("/**")
.excludePathPatterns("/user/login")
.excludePathPatterns("/user/logout");
}
}
addInterceptor
:需要一个实现
HandlerInterceptor
接口的拦截器实例
addPathPatterns
:用于设置拦截器的过滤路径规则;
addPathPatterns("/**")
对所有请求都拦截
excludePathPatterns
:用于设置不需要拦截的过滤规则
拦截器主要用途:进行用户登录状态的拦截,日志的拦截等