什么是SMTP?

SMTP全称为Simple Mail Transfer Protocol(简单邮件传输协议),它是一组用于从源地址到目的地址传输邮件的规范,通过它来控制邮件的中转方式。SMTP认证要求必须提供账号和密码才能登陆服务器,其设计目的在于避免用户受到垃圾邮件的侵扰。

什么是IMAP?

IMAP全称为Internet Message Access Protocol(互联网邮件访问协议),IMAP允许从邮件服务器上获取邮件的信息、下载邮件等。IMAP与POP类似,都是一种邮件获取协议。

什么是POP3?

POP3全称为Post Office Protocol 3(邮局协议),POP3支持客户端远程管理服务器端的邮件。POP3常用于“离线”邮件处理,即允许客户端下载服务器邮件,然后服务器上的邮件将会被删除。目前很多POP3的邮件服务器只提供下载邮件功能,服务器本身并不删除邮件,这种属于改进版的POP3协议。

IMAP和POP3协议有什么不同呢?

两者最大的区别在于,IMAP允许双向通信,即在客户端的操作会反馈到服务器上,例如在客户端收取邮件、标记已读等操作,服务器会跟着同步这些操作。而对于POP协议虽然也允许客户端下载服务器邮件,但是在客户端的操作并不会同步到服务器上面的,例如在客户端收取或标记已读邮件,服务器不会同步这些操作。

进阶知识

什么是JavaMailSender和JavaMailSenderImpl?

JavaMailSender和JavaMailSenderImpl 是Spring官方提供的集成邮件服务的接口和实现类,以简单高效的设计著称,目前是Java后端发送邮件和集成邮件服务的主流工具。

如何通过JavaMailSenderImpl发送邮件?

非常简单,直接在业务类注入JavaMailSenderImpl并调用send方法发送邮件。其中简单邮件可以通过SimpleMailMessage来发送邮件,而复杂的邮件(例如添加附件)可以借助MimeMessageHelper来构建MimeMessage发送邮件。例如:

    @Autowiredprivate JavaMailSenderImpl mailSender;public void sendMail() throws MessagingException {//简单邮件SimpleMailMessage simpleMailMessage = new SimpleMailMessage();simpleMailMessage.setFrom("admin@163.com");simpleMailMessage.setTo("socks@qq.com");simpleMailMessage.setSubject("Happy New Year");simpleMailMessage.setText("新年快乐!");mailSender.send(simpleMailMessage);//复杂邮件MimeMessage mimeMessage = mailSender.createMimeMessage();MimeMessageHelper messageHelper = new MimeMessageHelper(mimeMessage);messageHelper.setFrom("admin@163.com");messageHelper.setTo("socks@qq.com");messageHelper.setSubject("Happy New Year");messageHelper.setText("新年快乐!");messageHelper.addInline("doge.gif", new File("xx/xx/doge.gif"));messageHelper.addAttachment("work.docx", new File("xx/xx/work.docx"));mailSender.send(mimeMessage);}

为什么JavaMailSenderImpl 能够开箱即用 ?

所谓开箱即用其实就是基于官方内置的自动配置,翻看源码可知晓邮件自动配置类(MailSenderPropertiesConfiguration) 为上下文提供了邮件服务实例(JavaMailSenderImpl)。具体源码如下:

@Configuration
@ConditionalOnProperty(prefix = "spring.mail", name = "host")
class MailSenderPropertiesConfiguration {private final MailProperties properties;MailSenderPropertiesConfiguration(MailProperties properties) {this.properties = properties;}@Bean@ConditionalOnMissingBeanpublic JavaMailSenderImpl mailSender() {JavaMailSenderImpl sender = new JavaMailSenderImpl();applyProperties(sender);return sender;}

其中MailProperties是关于邮件服务器的配置信息,具体源码如下:

@ConfigurationProperties(prefix = "spring.mail")
public class MailProperties {private static final Charset DEFAULT_CHARSET = StandardCharsets.UTF_8;private String host;private Integer port;private String username;private String password;private String protocol = "smtp";private Charset defaultEncoding = DEFAULT_CHARSET;private Map<String, String> properties = new HashMap<>();
}

使用教程

一、开启邮件服务

登陆网易邮箱163,在设置中打开并勾选POP3/SMTP/IMAP服务,然后会得到一个授权码,这个邮箱和授权码将用作登陆认证。

二、配置邮件服务

首先咱们通过 Spring Initializr 创建工程springboot-send-mail,如图所示:

然后在pom.xml 引入web、thymeleaf 和spring-boot-starter-mail等相关依赖。例如:

    <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-thymeleaf</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-mail</artifactId></dependency><dependency><groupId>org.webjars</groupId><artifactId>webjars-locator-core</artifactId></dependency><dependency><groupId>org.webjars</groupId><artifactId>jquery</artifactId><version>3.3.1</version></dependency><dependency><groupId>org.webjars</groupId><artifactId>bootstrap</artifactId><version>3.3.7</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><scope>runtime</scope></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency></dependencies>

根据前面提到的配置项(MailProperties)填写相关配置信息,其中spring.mail.username 表示连接邮件服务器时认证的登陆账号,可以是普通的手机号或者登陆账号,并非一定是邮箱,为了解决这个问题,推荐大家在spring.mail. properties.from填写邮件发信人即真实邮箱。

然后在application.yml添加如下配置:

spring:mail:host: smtp.163.com #SMTP服务器地址username:  socks #登陆账号password: 123456 #登陆密码(或授权码)properties:from: socks@163.com #邮件发信人(即真实邮箱)thymeleaf:cache: falseprefix: classpath:/views/servlet:multipart:max-file-size: 10MB #限制单个文件大小max-request-size: 50MB #限制请求总量

透过前面的进阶知识,我们知道在发送邮件前,需要先构建 SimpleMailMessage或 MimeMessage 邮件信息类来填写邮件标题、邮件内容等信息,最后提交给JavaMailSenderImpl发送邮件,这样看起来没什么问题,也能实现既定目标,但在实际使用中会出现大量零散和重复的代码,还不便于保存邮件到数据库。

那么优雅的发送邮件应该是如何的呢?应该屏蔽掉这些构建信息和发送邮件的细节,不管是简单还是复杂邮件,都可以通过统一的API来发送邮件。例如:mailService.send(mailVo) 。

例如通过邮件信息类(MailVo) 来保存发送邮件时的邮件主题、邮件内容等信息 :

package com.hehe.vo;public class MailVo {private String id;//邮件idprivate String from;//邮件发送人private String to;//邮件接收人(多个邮箱则用逗号","隔开)private String subject;//邮件主题private String text;//邮件内容private Date sentDate;//发送时间private String cc;//抄送(多个邮箱则用逗号","隔开)private String bcc;//密送(多个邮箱则用逗号","隔开)private String status;//状态private String error;//报错信息@JsonIgnoreprivate MultipartFile[] multipartFiles;//邮件附件//省略GET&SET方法
}

三、发送邮件和附件

=========== 接下来正式介绍发送邮件的最核心逻辑 前方高能 =============

除了发送邮件之外,还包括检测邮件和保存邮件等操作,例如:

  • 检测邮件 checkMail(); 首先校验邮件收信人、邮件主题和邮件内容这些必填项,若为空则拒绝发送。

  • 发送邮件 sendMimeMail(); 其次通过MimeMessageHelper来解析MailVo并构建MimeMessage传输邮件。

  • 保存邮件 sendMimeMail(); 最后将邮件保存到数据库,便于统计和追查邮件问题。

本案例邮件业务类 MailService 的具体源码如下:

package com.hehe.service;/*** 邮件业务类 MailService*/
@Service
public class MailService {private Logger logger = LoggerFactory.getLogger(getClass());//提供日志类@Autowiredprivate JavaMailSenderImpl mailSender;//注入邮件工具类/*** 发送邮件*/public MailVo sendMail(MailVo mailVo) {try {checkMail(mailVo); //1.检测邮件sendMimeMail(mailVo); //2.发送邮件return saveMail(mailVo); //3.保存邮件} catch (Exception e) {logger.error("发送邮件失败:", e);//打印错误信息mailVo.setStatus("fail");mailVo.setError(e.getMessage());return mailVo;}}//检测邮件信息类private void checkMail(MailVo mailVo) {if (StringUtils.isEmpty(mailVo.getTo())) {throw new RuntimeException("邮件收信人不能为空");}if (StringUtils.isEmpty(mailVo.getSubject())) {throw new RuntimeException("邮件主题不能为空");}if (StringUtils.isEmpty(mailVo.getText())) {throw new RuntimeException("邮件内容不能为空");}}//构建复杂邮件信息类private void sendMimeMail(MailVo mailVo) {try {MimeMessageHelper messageHelper = new MimeMessageHelper(mailSender.createMimeMessage(), true);//true表示支持复杂类型mailVo.setFrom(getMailSendFrom());//邮件发信人从配置项读取messageHelper.setFrom(mailVo.getFrom());//邮件发信人messageHelper.setTo(mailVo.getTo().split(","));//邮件收信人messageHelper.setSubject(mailVo.getSubject());//邮件主题messageHelper.setText(mailVo.getText());//邮件内容if (!StringUtils.isEmpty(mailVo.getCc())) {//抄送messageHelper.setCc(mailVo.getCc().split(","));}if (!StringUtils.isEmpty(mailVo.getBcc())) {//密送messageHelper.setCc(mailVo.getBcc().split(","));}if (mailVo.getMultipartFiles() != null) {//添加邮件附件for (MultipartFile multipartFile : mailVo.getMultipartFiles()) {messageHelper.addAttachment(multipartFile.getOriginalFilename(), multipartFile);}}if (StringUtils.isEmpty(mailVo.getSentDate())) {//发送时间mailVo.setSentDate(new Date());messageHelper.setSentDate(mailVo.getSentDate());}mailSender.send(messageHelper.getMimeMessage());//正式发送邮件mailVo.setStatus("ok");logger.info("发送邮件成功:{}->{}", mailVo.getFrom(), mailVo.getTo());} catch (Exception e) {throw new RuntimeException(e);//发送失败}}//保存邮件private MailVo saveMail(MailVo mailVo) {//将邮件保存到数据库..return mailVo;}//获取邮件发信人public String getMailSendFrom() {return mailSender.getJavaMailProperties().getProperty("from");}
}

搞定了发送邮件最核心的业务逻辑,接下来咱们写一个简单页面用来发送邮件。

首先写好跟页面交互的控制器 MailController,具体源码如下:

@RestController
public class MailController {@Autowiredprivate MailService mailService;/*** 发送邮件的主界面*/@GetMapping("/")public ModelAndView index() {ModelAndView mv = new ModelAndView("mail/sendMail");//打开发送邮件的页面mv.addObject("from", mailService.getMailSendFrom());//邮件发信人return mv;}/*** 发送邮件*/@PostMapping("/mail/send")public MailVo sendMail(MailVo mailVo, MultipartFile[] files) {mailVo.setMultipartFiles(files);return mailService.sendMail(mailVo);//发送邮件和附件}
}

然后在/resources/views/mail目录新建sendMail.html,具体源码如下:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org"><head><meta charset="UTF-8"/><title>发送邮件</title><link th:href="@{/webjars/bootstrap/css/bootstrap.min.css}" rel="stylesheet" type="text/css"/><script th:src="@{/webjars/jquery/jquery.min.js}"></script><script th:href="@{/webjars/bootstrap/js/bootstrap.min.js}"></script></head><body>
<div class="col-md-6" style="margin:20px;padding:20px;border: #E0E0E0 1px solid;"><marquee behavior="alternate" onfinish="alert(12)" id="mq"onMouseOut="this.start();$('#egg').text('嗯 真听话!');"onMouseOver="this.stop();$('#egg').text('有本事放开我呀!');"><h5 id="egg">祝大家新年快乐!</h5><img id="doge" src="http://pics.sc.chinaz.com/Files/pic/faces/3709/7.gif" alt=""></marquee><form class="form-horizontal" id="mailForm"><div class="form-group"><label class="col-md-2 control-label">邮件发信人:</label><div class="col-md-6"><input class="form-control" id="from" name="from" th:value="${from}" readonly="readonly"></div></div><div class="form-group"><label class="col-md-2 control-label">邮件收信人:</label><div class="col-md-6"><input class="form-control" id="to" name="to" title="多个邮箱使用,隔开"></div></div><div class="form-group"><label class="col-md-2 control-label">邮件主题:</label><div class="col-md-6"><input class="form-control" id="subject" name="subject"></div></div><div class="form-group"><label class="col-md-2 control-label">邮件内容:</label><div class="col-md-6"><textarea class="form-control" id="text" name="text" rows="5"></textarea></div></div><div class="form-group"><label class="col-md-2 control-label">邮件附件:</label><div class="col-md-6"><input class="form-control" id="files" name="files" type="file" multiple="multiple"></div></div><div class="form-group"><label class="col-md-2 control-label">邮件操作:</label><div class="col-md-3"><a class="form-control btn btn-primary" onclick="sendMail()">发送邮件</a></div><div class="col-md-3"><a class="form-control btn btn-default" onclick="clearForm()">清空</a></div></div></form><script th:inline="javascript">var appCtx = [[${#request.getContextPath()}]];function sendMail() {var formData = new FormData($('#mailForm')[0]);$.ajax({url: appCtx + '/mail/send',type: "POST",data: formData,contentType: false,processData: false,success: function (result) {alert(result.status === 'ok' ? "发送成功!" : "你被Doge嘲讽了:" + result.error);},error: function () {alert("发送失败!");}});}function clearForm() {$('#mailForm')[0].reset();}setInterval(function () {var total = $('#mq').width();var width = $('#doge').width();var left = $('#doge').offset().left;if (left <= width / 2 + 20) {$('#doge').css('transform', 'rotateY(180deg)')}if (left >= total - width / 2 - 40) {$('#doge').css('transform', 'rotateY(-360deg)')}});</script>
</div>
</body>
</html>

四、测试发送邮件

如果是初学者,建议大家先下载源码,修改配置后运行工程,成功后再自己重新写一遍代码,这样有助于加深记忆。

启动工程并访问:http://localhost:8080 然后可以看到发送邮件的主界面如下:

然后填写你的小号邮箱,点击发送邮件,若成功则可以登陆小号邮箱查看邮件和刚才上传的附件。

文章推荐程序员效率:画流程图常用的工具程序员效率:整理常用的在线笔记软件远程办公:常用的远程协助软件,你都知道吗?51单片机程序下载、ISP及串口基础知识硬件:断路器、接触器、继电器基础知识

(转载)SpringBoot 发送邮件和附件相关推荐

  1. python发送邮件和附件

    发送邮件的时候,需要发送人,收件人,和一台邮件服务器,这里使用python发送一个邮件,主要需要引入smtplib和email库. 下面是源码,粘贴即可用: #!/usr/bin/env python ...

  2. SpringBoot 发送邮件功能实现

    背景 有个小伙伴问我你以前发邮件功能怎么弄的.然后我就给他找了个demo,正好在此也写一下,分享给大家. 理清痛点 发送邮件,大家可以想一下,坑的地方在哪? 我觉得是三个吧. 第一:邮件白名单问题. ...

  3. springboot发送邮件_SpringBoot发送邮件如何实现,SpringBoot发送邮件详解(附代码)...

    在很多的项目开发中,总会遇到消息通知的场景,比如某个广告主提交一个表单,我们要通知提醒运营人员及时查看. 消息通知的形式也有很多,比如:短信.邮件.app推送等,本文主要给大家描述一下邮件通知的形式, ...

  4. SpringBoot发送邮件(二)发送包含图片的邮件

    前言:使用发邮件这个功能不难,但是也有一些坑,下面我把开发邮件功能总结了一下分享给大家,同时为了避免篇幅过长,导致大家看的不仔细或看一半不想看了,我将这个功能细分了一下,写了好几篇供大家各取所需. S ...

  5. Python 使用 smtp ssl 模式 发送邮件与附件

    参考 :         发送邮件简单入门 ( 以qq邮箱,163邮箱为例 ) :https://blog.csdn.net/qq_38661599/article/details/81013834 ...

  6. SpringBoot发送邮件通过SendGrid服务平台(带动态模板)

    SpringBoot发送邮件通过SendGrid服务平台 一.开通API Keys 二.后台实现 一.开通API Keys 二.后台实现 加入maven依赖 <dependency>< ...

  7. python3发送qq邮件_python3通过qq邮箱发送邮件以及附件

    本文实例为大家分享了python3通过qq邮箱发送邮件以及附件的具体代码,供大家参考,具体内容如下ZLP免费资源网 开启qq邮箱的smtp服务ZLP免费资源网 ZLP免费资源网 代码:ZLP免费资源网 ...

  8. python3通过qq邮箱发送邮件以及附件

    本文实例为大家分享了python3通过qq邮箱发送邮件以及附件的具体代码,供大家参考,具体内容如下 开启qq邮箱的smtp服务 代码: 在学习过程中有什么不懂得可以加我的 python学习qun,85 ...

  9. springBoot发送邮件附件名称乱码的解决

    文章目录 问题 场景一 场景二 解决 完整源码 问题 场景一 场景二 解决 //java mail发邮件是附件名过长默认会被截断,附件名显示[tcmime.2475.2828.3041.bin],主动 ...

最新文章

  1. Python+OpenCV 图像处理系列(9)—— 图像的翻转和缩放插值
  2. 技术剖析 | Axonius为什么能获得 2019 RSAC创新大奖
  3. Java面向对象特征介绍
  4. 码牛安卓移动互联网高级开发正式课
  5. 三招助你轻松搞定超Diao氛围
  6. codova添加android慢_从 0 开始学 Linux 内核之 android 内核栈溢出 ROP 利用
  7. ACM投稿ccs concepts查询The ACM Computing Classification System
  8. ldDWR出现“例外被抛出且未被接住 解决方法
  9. 微信小程序实现图片上传功能
  10. matlab计算abc三相短路电流_变频电源的安装及单相/三相变频电源电流计算公
  11. Foobar2000 CUI界面foobox开发已经接近尾声
  12. 坚定推动DDD落地的企业,70%代码效率翻倍了!
  13. 浮窗---创建Activity浮窗(可拖动)
  14. 大学计算机作业互评评语简短,同学作业互评评语
  15. 北京国际学校ib成绩排名如何?
  16. [活动预告] Substrate 中的 IBC 跨链模块技术分享 Substrate-ibc
  17. 华为怎么改输入法皮肤_华为输入法
  18. bootstrap实战--微金所项目(轮播图1)
  19. 敢问路在何方 路在脚下
  20. 一台Linux服务器上挂载另一台Linux服务器文件系统的方法

热门文章

  1. LOAM_velodyne学习(二)
  2. python实时连接oracle_Python连接Oracle
  3. python win+r时不成功_Win与R(不使用Anaconda的情况下)
  4. closewait一直不释放_机床为什么要释放应力?怎么释放应力才好?
  5. php 完美分页,php完美分页类程序
  6. mysql8创建用户并授权_新特性解读 | 从 wireshark 看 MySQL 8.0 加密连接
  7. 求1到100中9的个数
  8. 江西计算机一级考试教程,江西省2019年下半年计算机一级考试复习教程:计算机基础及MS Office应用上机指导...
  9. Luogu 4721 【模板】分治 FFT
  10. 3ds Max 2018 在安装后无法启动或出现不稳定