写在前面 我们都知道,如果系统出现预警,或者有一些监控需求,我们可以通过发送短信或者邮件来进行通知,本篇文章呢,我就打算来讲解一下SpringBoot的邮件服务。我们都知道发送邮件应该是网站的必备功能之一,什么注册验证,忘记密码或者是给用户发送营销信息。以前我们会使用 JavaMail 相关 api 来写发送邮件的相关代码,后来 Spring 推出了 JavaMailSender 更加简化了邮件发送的过程,在之后 Spring Boot 对此进行了封装就有了现在的 spring-boot-starter-mail ,如果你看了我前面的文章的话,就会知道,SpringBoot把大部分的需求封装成了一个个场景启动器,而mail也就是相应的场景启动器。
了解邮件服务 经常出现和邮件相关的协议是SMTP、IMAP和POP3,所以在这里我们首先来认识了解这三个协议。
SMTP全称为Simple Mail Transfer Protocol(简单邮件传输协议),它是一组用于从源地址到目的地址传输邮件的规范,通过它来控制邮件的中转方式。SMTP认证要求必须提供账号和密码才能登陆服务器,其设计目的在于避免用户受到垃圾邮件的侵扰。
IMAP全称为Internet Message Access Protocol(互联网邮件访问协议),IMAP允许从邮件服务器上获取邮件的信息、下载邮件等。IMAP与POP类似,都是一种邮件获取协议。
POP3全称为Post Office Protocol 3(邮局协议),POP3支持客户端远程管理服务器端的邮件。POP3常用于“离线”邮件处理,即允许客户端下载服务器邮件,然后服务器上的邮件将会被删除。目前很多POP3的邮件服务器只提供下载邮件功能,服务器本身并不删除邮件,这种属于改进版的POP3协议。
那么问题来了,IMAP和POP3协议有什么不同呢?两者最大的区别在于,IMAP允许双向通信,即在客户端的操作会反馈到服务器上,例如在客户端收取邮件、标记已读等操作,服务器会跟着同步这些操作。而对于POP协议虽然也允许客户端下载服务器邮件,但是在客户端的操作并不会同步到服务器上面的,例如在客户端收取或标记已读邮件,服务器不会同步这些操作。
SpringBoot相关类 SpringBoot中针对邮件服务的两个工具类是,JavaMailSender和JavaMailSenderImpl,它们是Spring官方提供的集成邮件服务的接口和实现类,以简单高效的设计著称,目前是Java后端发送邮件和集成邮件服务的主流工具。那如何通过JavaMailSenderImpl发送邮件?非常简单,直接在业务类注入JavaMailSenderImpl并调用send方法发送邮件。其中简单邮件可以通过SimpleMailMessage来发送邮件,而复杂的邮件(例如添加附件)可以借助MimeMessageHelper来构建MimeMessage发送邮件。
我们不难理解,SpringBoot对于邮件服务能做到开箱即用,其实就是基于官方内置的自动配置,翻看源码可知晓邮件自动配置类(MailSenderPropertiesConfiguration) 为上下文提供了邮件服务实例(JavaMailSenderImpl)。
具体教程 配置 首先我们创建一个新的项目,只要包含最基本的web场景就可以了,然后我们在pom.xml中引入依赖就可以了,依赖如下:
1 2 3 4 5 6 <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-mail</artifactId> </dependency> </dependencies>
接着,我们在application.properties主配置文件中对mail进行相关的配置,配置内容如下,我做了相关注释
1 2 3 4 5 spring.mail.host=smtp.163.com spring.mail.username=amazing spring.mail.password=xxxxxx spring.mail.default-encoding=UTF-8 mail.fromMail.address=amazing@163.com
上面的邮箱服务器的地址,我这里放出一下常用有限发邮箱服务器地址
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 QQ邮箱(mail.qq.com) POP3服务器地址:pop.qq.com(端口:110) SMTP服务器地址:smtp.qq.com(端口:25) SMTP服务器需要身份验证。 网易邮箱(163.com): POP3服务器地址:pop.163.com(端口:110) SMTP服务器地址:smtp.163.com(端口:25) 谷歌邮箱(google.com): POP3服务器地址:pop.gmail.com(SSL启用端口:995) SMTP服务器地址:smtp.gmail.com(SSL启用端口:587) 阿里云邮箱(mail.aliyun.com): POP3服务器地址:pop3.aliyun.com(SSL加密端口:995;非加密端口:110) SMTP服务器地址:smtp.aliyun.com(SSL加密端口:465;非加密端口:25) IMAP服务器地址:imap.aliyun.com(SSL加密端口:993;非加密端口:143) 新浪邮箱(sina.com): POP3服务器地址:pop3.sina.com.cn(端口:110) SMTP服务器地址:smtp.sina.com.cn(端口:25)
简单使用 这样就完成了我们SpringBoot使用邮件服务的基本配置,那么接下来我们简单使用一下,首先编写Service,目录结构如下
1 2 3 public interface MailService { public void sendSimpleMail (String to, String subject, String content) ; }
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 @Component public class MailServiceImpl implements MailService { private final Logger logger = LoggerFactory.getLogger(this .getClass()); @Autowired private JavaMailSender mailSender; @Value("${mail.fromMail.addr}") private String from; @Override public void sendSimpleMail (String to, String subject, String content) { SimpleMailMessage message = new SimpleMailMessage(); message.setFrom(from); message.setTo(to); message.setSubject(subject); message.setText(content); try { mailSender.send(message); logger.info("简单邮件已经发送。" ); } catch (Exception e) { logger.error("发送简单邮件时发生异常!" , e); } } }
编写 test 类进行测试,至此一个简单的文本发送就完成了。
丰富邮件内容 但是在正常使用的过程中,我们通常在邮件中加入图片或者附件来丰富邮件的内容,下面讲介绍如何使用 Spring Boot 来发送丰富的邮件。
发送 html 格式邮件 其它都不变在 MailService 添加 sendHtmlMail 方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 public void sendHtmlMail (String to, String subject, String content) { MimeMessage message = mailSender.createMimeMessage(); try { MimeMessageHelper helper = new MimeMessageHelper(message, true ); helper.setFrom(from); helper.setTo(to); helper.setSubject(subject); helper.setText(content, true ); mailSender.send(message); logger.info("html邮件发送成功" ); } catch (MessagingException e) { logger.error("发送html邮件时发生异常!" , e); } }
在测试类中构建 html 内容,测试发送
1 2 3 4 5 6 7 8 9 @Test public void testHtmlMail () throws Exception { String content="<html>\n" + "<body>\n" + " <h3>hello world ! 这是一封Html邮件!</h3>\n" + "</body>\n" + "</html>" ; MailService.sendHtmlMail("xxxxxx@163.com" ,"test simple mail" ,content); }
发送带附件的邮件 还是老样子,在 MailService 添加 sendAttachmentsMail 方法。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 public void sendAttachmentsMail (String to, String subject, String content, String filePath) { MimeMessage message = mailSender.createMimeMessage(); try { MimeMessageHelper helper = new MimeMessageHelper(message, true ); helper.setFrom(from); helper.setTo(to); helper.setSubject(subject); helper.setText(content, true ); FileSystemResource file = new FileSystemResource(new File(filePath)); String fileName = filePath.substring(filePath.lastIndexOf(File.separator)); helper.addAttachment(fileName, file); mailSender.send(message); logger.info("带附件的邮件已经发送。" ); } catch (MessagingException e) { logger.error("发送带附件的邮件时发生异常!" , e); } }
添加多个附件可以使用多条 helper.addAttachment(fileName, file)
,然后在测试类中添加测试方法
1 2 3 4 5 @Test public void sendAttachmentsMail () { String filePath="e:\\tmp\\application.log" ; mailService.sendAttachmentsMail("xxxxx@163.com" , "主题:带附件的邮件" , "有附件,请查收!" , filePath); }
发送带静态资源的邮件 邮件中的静态资源一般就是指图片,在 MailService 添加 sendAttachmentsMail 方法。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 public void sendInlineResourceMail (String to, String subject, String content, String rscPath, String rscId) { MimeMessage message = mailSender.createMimeMessage(); try { MimeMessageHelper helper = new MimeMessageHelper(message, true ); helper.setFrom(from); helper.setTo(to); helper.setSubject(subject); helper.setText(content, true ); FileSystemResource res = new FileSystemResource(new File(rscPath)); helper.addInline(rscId, res); mailSender.send(message); logger.info("嵌入静态资源的邮件已经发送。" ); } catch (MessagingException e) { logger.error("发送嵌入静态资源的邮件时发生异常!" , e); } }
在测试类中添加测试方法
1 2 3 4 5 6 7 8 @Test public void sendInlineResourceMail () { String rscId = "neo006" ; String content="<html><body>这是有图片的邮件:<img src=\'cid:" + rscId + "\' ></body></html>" ; String imgPath = "C:\\Users\\summer\\Pictures\\favicon.png" ; mailService.sendInlineResourceMail("XXXXX@163.com" , "主题:这是有图片的邮件" , content, imgPath, rscId); }
添加多个图片可以使用多条 <img src='cid:" + rscId + "' >
和 helper.addInline(rscId, res)
来实现
邮件系统 上面发送邮件的基础服务就这些了,但是如果我们要做成一个邮件系统的话还需要考虑以下几个问题,首先是邮件模板的问题,我们会经常收到类似这样的邮件 其中只有 neo 这个用户名在变化,其它邮件内容均不变,如果每次发送邮件都需要手动拼接的话会不够优雅,并且每次模板的修改都需要改动代码的话也很不方便,因此对于这类邮件需求,都建议做成邮件模板来处理。模板的本质很简单,就是在模板中替换变化的参数,转换为 html 字符串即可,这里以thymeleaf为例来演示,第一步当然是导入thymeleaf的包啦。
1 2 3 4 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-thymeleaf</artifactId> </dependency>
然后在在 resorces/templates
下创建 emailTemplate.html
1 2 3 4 5 6 7 8 9 10 11 <!DOCTYPE html> <html lang="zh" xmlns:th="http://www.thymeleaf.org" > <head> <meta charset="UTF-8" /> <title>Title</title> </head> <body> 您好,这是验证邮件,请点击下面的链接完成验证,<br/> <a href="#" th:href="@{ http://www.xxx.com/neo/{id}(id=${id}) }">激活账号</a> </body> </html>
1 2 3 4 5 6 7 8 9 @Test public void sendTemplateMail () { Context context = new Context(); context.setVariable("id" , "006" ); String emailContent = templateEngine.process("emailTemplate" , context); mailService.sendHtmlMail("ityouknow@126.com" ,"主题:这是模板邮件" ,emailContent); }
当然啦,这里要强调一点的是,我们在实现邮箱服务的时候,因为各种原因,总会有邮件发送失败的情况,比如:邮件发送过于频繁、网络异常等。在出现这种情况的时候,我们一般会考虑重新重试发送邮件,会分为以下几个步骤来实现:
接收到发送邮件请求,首先记录请求并且入库。
调用邮件发送接口发送邮件,并且将发送结果记录入库。
启动定时系统扫描时间段内,未发送成功并且重试次数小于3次的邮件,进行再次发送
最后 很多时候邮件发送并不是我们主业务必须关注的结果,比如通知类、提醒类的业务可以允许延时或者失败。这个时候可以采用异步的方式来发送邮件,加快主交易执行速度,在实际项目中可以采用MQ发送邮件相关参数,监听到消息队列之后启动发送邮件。
Author:
DengBoCong
Permalink:
http://dengbocong.cn/Spring-Boot/f76f168ee368/
License:
Licensed under the Apache License, Version 2.0 (the "License")
Slogan:
Stay hungry, Stay foolish.