异步任务

异步测试

  • 异步处理还是非常常用的,比如我们在网站上发送邮件,后台会去发送邮件,此时前台会造成响应不动,直到邮件发送完毕,响应才会成功,所以我们一般会采用多线程的方式去处理这些任务
  • 编写方法,假装正在处理数据,使用线程设置一些延时,模拟同步等待的情况;
  • 编写一个Service层方法
1
2
3
4
5
6
7
8
9
10
11
12
@Service
public class AsyncService {

public void hello(){
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("业务进行中....");
}
}
  • 编写Controller层进行请求测试
1
2
3
4
5
6
7
8
9
10
11
@RestController
public class AsyncController {
@Autowired
AsyncService asyncService;

@GetMapping("/hello")
public String hello(){
asyncService.hello();
return "success";
}
}
  • 访问http://localhost:8080/hello进行测试,3秒后出现success,这是同步等待的情况

@Async

  • 问题:我们如果想让用户直接得到消息,就在后台使用多线程的方式进行处理即可,但是每次都需要自己手动去编写多线程的实现的话,太麻烦了,我们只需要用一个简单的办法,在我们的方法上加一个简单的注解即可,如下
1
2
3
4
5
6
7
8
9
10
11
12
13
@Service
public class AsyncService {
// 告诉Spring这是一个异步方法
@Async
public void hello(){
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("业务进行中....");
}
}
  • SpringBoot就会自己开一个线程池,进行调用!但是要让这个注解生效,我们还需要在主程序上添加一个注解@EnableAsync ,开启异步注解功能;
1
2
3
4
5
6
7
8
@EnableAsync //开启异步注解功能
@SpringBootApplication
public class SpringbootTaskApplication {

public static void main(String[] args) {
SpringApplication.run(SpringbootTaskApplication.class, args);
}
}
  • 重启测试,网页瞬间响应,后台代码依旧执行!

定时任务

  • 项目开发中经常需要执行一些定时任务,比如需要在每天凌晨的时候,分析一次前一天的日志信息,Spring为我们提供了异步执行任务调度的方式,提供了两个接口。
    • TaskExecutor接口
    • TaskScheduler接口
  • 两个注解:
    • @EnableScheduling:开启定时功能的注解,配置在启动类
    • @Scheduled:什么时候执行

cron表达式

  • Cron表达式是一个字符串,字符串以5或6个空格隔开,分为6或7个域,每一个域代表一个含义,Cron有如下两种语法格式:
1
2
Seconds Minutes Hours DayofMonth Month DayofWeek Year
Seconds Minutes Hours DayofMonth Month DayofWeek
  • corn从左到右(用空格隔开):秒 分 小时 月份中的日期 月份 星期中的日期 年份

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
10/2 * * * * ?   表示每2秒 执行任务
10 0/2 * * * ? 表示每2分钟 执行任务
10 0 2 1 * ? 表示在每月的1日的凌晨2点调整任务
20 15 10 ? * MON-FRI 表示周一到周五每天上午10:15执行作业
30 15 10 ? 6L 2002-2006 表示2002-2006年的每个月的最后一个星期五上午10:15执行作
40 0 10,14,16 * * ? 每天上午10点,下午2点,4
50 0/30 9-17 * * ? 朝九晚五工作时间内每半小时
60 0 12 ? * WED 表示每个星期三中午12
70 0 12 * * ? 每天中午12点触发
80 15 10 ? * * 每天上午10:15触发
90 15 10 * * ? 每天上午10:15触发
100 15 10 * * ? 每天上午10:15触发
110 15 10 * * ? 2005 2005年的每天上午10:15触发
120 * 14 * * ? 在每天下午2点到下午2:59期间的每1分钟触发
130 0/5 14 * * ? 在每天下午2点到下午2:55期间的每5分钟触发
140 0/5 14,18 * * ? 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
150 0-5 14 * * ? 在每天下午2点到下午2:05期间的每1分钟触发
160 10,44 14 ? 3 WED 每年三月的星期三的下午2:102:44触发
170 15 10 ? * MON-FRI 周一至周五的上午10:15触发
180 15 10 15 * ? 每月15日上午10:15触发
190 15 10 L * ? 每月最后一日的上午10:15触发
200 15 10 ? * 6L 每月的最后一个星期五上午10:15触发
210 15 10 ? * 6L 2002-2005 2002年至2005年的每月的最后一个星期五上午10:15触发
(22)0 15 10 ? * 6#3 每月的第三个星期五上午10:15触发

定时测试

  • 创建一个ScheduledService
1
2
3
4
5
6
7
8
9
10
@Service
public class ScheduledService {
// 在特定时间执行
//秒 分 时 日 月 周几
// 明天14点52分 执行一次
@Scheduled(cron = "0 52 14 * * ?")
public void hello(){
System.out.println("hello.....");
}
}
  • 写完定时任务之后,我们需要在主程序上增加@EnableScheduling 开启定时任务功能
1
2
3
4
5
6
7
8
9
10
@EnableAsync //开启异步注解功能
@EnableScheduling //开启基于注解的定时任务
@SpringBootApplication
public class SpringbootTaskApplication {

public static void main(String[] args) {
SpringApplication.run(SpringbootTaskApplication.class, args);
}

}
  • 测试结果,在14:52分执行方法输出结果

邮件任务

邮件配置

  • 邮件发送,在我们的日常开发中,也非常的多,Springboot也帮我们做了支持
    • 邮件发送需要引入spring-boot-start-mail
    • SpringBoot 自动配置MailSenderAutoConfiguration
    • 定义MailProperties内容,配置在application.yml
    • 自动装配JavaMailSender
    • 测试邮件发送
1
2
3
4
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-mail</artifactId>
</dependency>
  • Alt键看它引入的父依赖,可以看到 jakarta.mail
1
2
3
4
5
6
<dependency>
<groupId>com.sun.mail</groupId>
<artifactId>jakarta.mail</artifactId>
<version>1.6.5</version>
<scope>compile</scope>
</dependency>
  • 搜索查看自动配置类:MailSenderAutoConfiguration

  • 这个类中存在JavaMailSenderImpl

  • 看下配置文件类MailProperties
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@ConfigurationProperties(
prefix = "spring.mail"
)
public class MailProperties {
private static final Charset DEFAULT_CHARSET;
private String host;
private Integer port;
private String username;
private String password;
private String protocol = "smtp";
private Charset defaultEncoding;
private Map<String, String> properties;
private String jndiName;
}

邮件测试

  • 在·application.properties中增加配置
1
2
3
4
5
spring.mail.username=2312057536@qq.com
spring.mail.password=你的qq授权码
spring.mail.host=smtp.qq.com
# qq需要配置ssl
spring.mail.properties.mail.smtp.ssl.enable=true
  • 在SpringBoot单元测试中发送一个简单的邮件
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
@SpringBootTest
class SpringbootTaskApplicationTests {
@Autowired
JavaMailSenderImpl mailSender;
@Test
void contextLoads() {
//1. 简单的邮件
SimpleMailMessage mailMessage = new SimpleMailMessage();
mailMessage.setSubject("主题");
mailMessage.setText("邮件内容");
mailMessage.setTo("xiaoliblog123@163.com");
mailMessage.setFrom("2312057536@qq.com");
mailSender.send(mailMessage);
}
@Test
void contextLoads2() throws MessagingException {
//2. 一个复杂的邮件
MimeMessage mimeMessage = mailSender.createMimeMessage();
// 组装
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage, true);

helper.setSubject("验证码通知");
helper.setText("<b style='color:red'>您的验证码为1234</b>",true);

//发送附件
helper.addAttachment("1.jpg",new File("C:\\1.jpg"));
helper.addAttachment("2.jpg",new File("C:\\2.jpg"));

helper.setTo("2312057536@qq.com");
helper.setFrom("2312057536@qq.com");

mailSender.send(mimeMessage);
}

}