prefácio
Em nosso trabalho, as tarefas de processamento assíncrono são frequentemente usadas. Por exemplo, quando enviamos e-mails no site, o plano de fundo enviará e-mails. Neste momento, a recepção fará com que a resposta permaneça estática. A resposta não será bem-sucedida até que o e-mail é enviado, então geralmente essas tarefas são tratadas de maneira multi-thread. Há também algumas tarefas agendadas, como a necessidade de analisar as informações de log do dia anterior todas as manhãs. Há também o envio de e-mails. O antecessor do WeChat também é um serviço de e-mail? Como essas coisas são alcançadas? Na verdade, SpringBoot nos fornece suporte correspondente. É muito simples para nós usá-lo. Nós só precisamos habilitar algum suporte de anotação e configurar alguns arquivos de configuração! Então vamos dar uma olhada~
tarefa assíncrona
1. Crie um pacote de serviços
2. Crie uma classe AsyncService
O processamento assíncrono ainda é muito comum. Por exemplo, quando enviamos e-mails no site, o plano de fundo enviará os e-mails. Neste momento, a recepção fará com que a resposta não seja movida. A resposta não será bem-sucedida até que o e-mail seja enviado , portanto, geralmente usamos métodos multiencadeados para lidar com essas tarefas.
Escreva um método que finja estar processando dados, use threads para definir alguns atrasos e simule esperas síncronas;
@Service
public class AsyncService {
public void hello(){
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("业务进行中....");
}
}
3. Escreva o pacote do controlador
4. Escreva a classe AsyncController
Vamos escrever um Controller para testá-lo
@RestController
public class AsyncController {
@Autowired
AsyncService asyncService;
@GetMapping("/hello")
public String hello(){
asyncService.hello();
return "success";
}
}
5. Visite http://localhost:8080/hello para testar, e o sucesso ocorre após 3 segundos, que é a situação de espera síncrona.
Pergunta : Se quisermos que o usuário receba a mensagem diretamente, podemos usar o processamento multithread em segundo plano, mas é muito problemático escrever manualmente a implementação multithread toda vez. , adicione uma anotação simples ao nosso método, como segue:
6. Adicione a anotação @Async ao método hello;
//告诉Spring这是一个异步方法
@Async
public void hello(){
try {
Thread.sleep(3000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("业务进行中....");
}
O SpringBoot abrirá um pool de threads sozinho e o chamará! Mas para que essa anotação tenha efeito, também precisamos adicionar uma anotação @EnableAsync ao programa principal para habilitar a função de anotação assíncrona;
@EnableAsync //开启异步注解功能
@SpringBootApplication
public class SpringbootTaskApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootTaskApplication.class, args);
}
}
7. Reinicie o teste, a página da web responde instantaneamente e o código em segundo plano ainda é executado!
tarefa cronometrada
No desenvolvimento de projetos, muitas vezes é necessário realizar algumas tarefas agendadas. Por exemplo, é necessário analisar as informações de log do dia anterior no início da manhã de cada dia. O Spring nos fornece uma maneira de executar o agendamento de tarefas de forma assíncrona e fornece duas interfaces.
- Interface do TaskExecutor
- Interface do TaskScheduler
Duas anotações:
- @EnableScheduling
- @Agendado
expressão cron:
Etapas do teste:
1. Crie um Serviço Agendado
Existe um método hello em nós, ele precisa ser executado regularmente, como lidar com isso?
@Service
public class ScheduledService {
//秒 分 时 日 月 周几
//0 * * * * MON-FRI
//注意cron表达式的用法;
@Scheduled(cron = "0 * * * * 0-7")
public void hello(){
System.out.println("hello.....");
}
}
2. Depois de escrever a tarefa agendada aqui, precisamos adicionar @EnableScheduling ao programa principal para habilitar a função de tarefa agendada
@EnableAsync //开启异步注解功能
@EnableScheduling //开启基于注解的定时任务
@SpringBootApplication
public class SpringbootTaskApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootTaskApplication.class, args);
}
}
3. Vamos aprender mais sobre expressões cron;
https://www.bejson.com/othertools/cron/
4. Expressões comumente usadas
(1)0/2 * * * * ? 表示每2秒 执行任务
(1)0 0/2 * * * ? 表示每2分钟 执行任务
(1)0 0 2 1 * ? 表示在每月的1日的凌晨2点调整任务
(2)0 15 10 ? * MON-FRI 表示周一到周五每天上午10:15执行作业
(3)0 15 10 ? 6L 2002-2006 表示2002-2006年的每个月的最后一个星期五上午10:15执行作
(4)0 0 10,14,16 * * ? 每天上午10点,下午2点,4点
(5)0 0/30 9-17 * * ? 朝九晚五工作时间内每半小时
(6)0 0 12 ? * WED 表示每个星期三中午12点
(7)0 0 12 * * ? 每天中午12点触发
(8)0 15 10 ? * * 每天上午10:15触发
(9)0 15 10 * * ? 每天上午10:15触发
(10)0 15 10 * * ? 每天上午10:15触发
(11)0 15 10 * * ? 2005 2005年的每天上午10:15触发
(12)0 * 14 * * ? 在每天下午2点到下午2:59期间的每1分钟触发
(13)0 0/5 14 * * ? 在每天下午2点到下午2:55期间的每5分钟触发
(14)0 0/5 14,18 * * ? 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
(15)0 0-5 14 * * ? 在每天下午2点到下午2:05期间的每1分钟触发
(16)0 10,44 14 ? 3 WED 每年三月的星期三的下午2:10和2:44触发
(17)0 15 10 ? * MON-FRI 周一至周五的上午10:15触发
(18)0 15 10 15 * ? 每月15日上午10:15触发
(19)0 15 10 L * ? 每月最后一日的上午10:15触发
(20)0 15 10 ? * 6L 每月的最后一个星期五上午10:15触发
(21)0 15 10 ? * 6L 2002-2005 2002年至2005年的每月的最后一个星期五上午10:15触发
(22)0 15 10 ? * 6#3 每月的第三个星期五上午10:15触发
tarefa de correio
Há também muitos envios de e-mail em nosso desenvolvimento diário, e o Springboot também nos apoia.
-
O envio de e-mail precisa introduzir o spring-boot-start-mail
-
SpringBoot configura automaticamente MailSenderAutoConfiguration
-
Defina o conteúdo de MailProperties, configurado em application.yml
-
Autowire JavaMailSender
-
Teste o envio de e-mail
teste:
1. Introduzir dependências pom
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-mail</artifactId>
</dependency>
Clique no código-fonte para ver as dependências que ele introduz, você pode ver jakarta.mail
<dependency>
<groupId>com.sun.mail</groupId>
<artifactId>jakarta.mail</artifactId>
<version>1.6.4</version>
<scope>compile</scope>
</dependency>
2. Visualize a classe de configuração automática: MailSenderAutoConfiguration
Existe um bean nesta classe, JavaMailSenderImpl
Em seguida, olhamos para o arquivo de configuração
@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;
}
Obtenha o código de autorização: Configurações na caixa de correio QQ -> Conta -> Habilitar serviços pop3 e smtp
3. Arquivo de configuração:
[email protected]
spring.mail.password=iarkclztxfanbdjc
spring.mail.host=smtp.qq.com
# 开启加密验证
spring.mail.properties.mail.smtp.ssl.enable=true
4. Teste de unidade de mola
package com.kwok;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.mail.SimpleMailMessage;
import org.springframework.mail.javamail.JavaMailSenderImpl;
import org.springframework.mail.javamail.MimeMessageHelper;
import javax.mail.MessagingException;
import javax.mail.internet.MimeMessage;
import java.io.File;
@SpringBootTest
class Springboot09TasktestApplicationTests {
@Autowired
JavaMailSenderImpl mailSender;
@Test
void contextLoads() {
// 一个简单的邮件~
SimpleMailMessage mailMessage = new SimpleMailMessage();
mailMessage.setSubject("HelloEmail~");
mailMessage.setText("springboot测试邮件发送");
mailMessage.setTo("[email protected]");
mailMessage.setFrom("[email protected]");
mailSender.send(mailMessage);
}
@Test
void contextLoads2() throws MessagingException {
// 一个复杂的邮件~
MimeMessage mimeMessage = mailSender.createMimeMessage();
// 组装~
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage,true);
helper.setSubject("HelloEmail2~");
helper.setText("<h1 style='color:blue'>springboot测试复杂邮件发送</h1>",true);
// 附件
helper.addAttachment("1.jpg",new File("E:\\a-IdeaProjects\\SpringBootStudy\\springboot-09-tasktest\\src\\main\\resources\\static\\head.jpg"));
helper.addAttachment("2.jpg",new File("E:\\a-IdeaProjects\\SpringBootStudy\\springboot-09-tasktest\\src\\main\\resources\\static\\head.jpg"));
helper.setTo("[email protected]");
helper.setFrom("[email protected]");
mailSender.send(mimeMessage);
}
/**
*
* @param html
* @param subject
* @param text
* @throws MessagingException
* @Author hokwok
*/
// 封装成方法
public void sendMail(Boolean html,String subject,String text) throws MessagingException {
// 一个复杂的邮件~
MimeMessage mimeMessage = mailSender.createMimeMessage();
// 组装~
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage,html);
helper.setSubject(subject);
helper.setText(text,true);
// 附件
helper.addAttachment("1.jpg",new File("E:\\a-IdeaProjects\\SpringBootStudy\\springboot-09-tasktest\\src\\main\\resources\\static\\head.jpg"));
helper.addAttachment("2.jpg",new File("E:\\a-IdeaProjects\\SpringBootStudy\\springboot-09-tasktest\\src\\main\\resources\\static\\head.jpg"));
helper.setTo("[email protected]");
helper.setFrom("[email protected]");
mailSender.send(mimeMessage);
}
}
Verifique a caixa de correio, o e-mail foi recebido com sucesso!
Nós só precisamos usar o Thymeleaf para combinar os front-ends e back-ends para desenvolver nossas próprias funções de envio e recebimento de e-mail do site!