本文是vhr系列的第十二篇,项目地址 https://github.com/lenve/vhr
邮件发送也是一个老生常谈的问题了,代码虽然简单,但是许多小伙伴对过程不太理解,所以还是打算和各位小伙伴聊聊这个话题。
我们经常会听到各种各样的邮件协议,比如SMTP、POP3、IMAP,那么这些协议有什么作用,有什么区别?我们先来讨论一下这个问题。
SMTP是一个基于TCP/IP的应用层协议,江湖地位有点类似于HTTP,SMTP服务器默认监听的端口号为25。看到这里,小伙伴们可能会想到既然SMTP协议是基于TCP/IP的应用层协议,那么我是不是也可以通过Socket发送一封邮件呢?回答是肯定的。
生活中我们投递一封邮件要经过如下几个步骤:
1.深圳的小王先将邮件投递到深圳的邮局
2.深圳的邮局将邮件运送到上海的邮局
3.上海的小张来邮局取邮件
这是一个缩减版的生活中邮件发送过程。这三个步骤可以分别对应我们的邮件发送过程,假设从aaa@qq.com发送邮件到111@163.com:
1.aaa@qq.com先将邮件投递到腾讯的邮件服务器
2.腾讯的邮件服务器将我们的邮件投递到网易的邮件服务器
3.111@163.com登录网易的邮件服务器查看邮件
邮件投递大致就是这个过程,这个过程就涉及到了多个协议,我们来分别看一下。
SMTP协议全称为Simple Mail Transfer Protocol,译作简单邮件传输协议,它定义了邮件客户端软件于SMTP服务器之间,以及SMTP服务器与SMTP服务器之间的通信规则。也就是说aaa@qq.com用户先将邮件投递到腾讯的SMTP服务器这个过程就使用了SMTP协议,然后腾讯的SMTP服务器将邮件投递到网易的SMTP服务器这个过程也依然使用了SMTP协议,SMTP服务器就是用来收邮件。而POP3协议全称为Post Office Protocol,译作邮局协议,它定义了邮件客户端与POP3服务器之间的通信规则,那么该协议在什么场景下会用到呢?当邮件到达网易的SMTP服务器之后,111@163.com用户需要登录服务器查看邮件,这个时候就该协议就用上了:邮件服务商都会为每一个用户提供专门的邮件存储空间,SMTP服务器收到邮件之后,就将邮件保存到相应用户的邮件存储空间中,如果用户要读取邮件,就需要通过邮件服务商的POP3邮件服务器来完成。最后,可能也有小伙伴们听说过IMAP协议,这个协议是对POP3协议的扩展,功能更强,作用类似,这里不再赘述。
首先我们需要先登录QQ邮箱网页版,点击上方的设置按钮:
然后点击账户选项卡:
在账户选项卡中找到开启POP3/SMTP选项,如下:
点击开启,开启相关功能,开启过程需要手机号码验证,按照步骤操作即可,不赘述。开启成功之后,即可获取一个授权码,将该号码保存好,一会使用。
然后我们需要JavaxMail这个jar包,小伙伴可以直接去Maven中央仓库下载,这里不再赘述。
如果我们只发送一个简单的文本,发送方式就比较简单,整个过程可以分为三步如下:
第一步:构造SMTP邮件服务器的基本环境
Properties properties = new Properties(); properties.setProperty("mail.host", "smtp.qq.com"); properties.setProperty("mail.transport.protocol", "smtp"); properties.setProperty("mail.smtp.auth", "true"); properties.setProperty("mail.smtp.socketFactory.class", "javax.net.ssl.SSLSocketFactory"); properties.setProperty("mail.smtp.port", "465"); Session session = Session.getDefaultInstance(properties); session.setDebug(true);
第二步:构造邮件
MimeMessage mimeMessage = new MimeMessage(session); mimeMessage.addRecipients(Message.RecipientType.TO, "111@qq.com");//设置收信人 mimeMessage.addRecipients(Message.RecipientType.CC, "222@qq.com");//抄送 mimeMessage.setFrom("1510161612@qq.com");//邮件发送人 mimeMessage.setSubject("测试邮件主题");//邮件主题 mimeMessage.setContent("Hello,这是一封测试邮件", "text/html;charset=utf-8");//正文
第三步:发送邮件
Transport transport = session.getTransport(); transport.connect("smtp.qq.com", "333@qq.com", "刚刚申请到的授权码"); transport.sendMessage(mimeMessage, mimeMessage.getAllRecipients());//发送邮件,第二个参数为收件人 transport.close();
发送复杂邮件,第一步和第三步也是一样的,只有第二步构造邮件的过程比较麻烦,那么接下来给小伙伴们演示一个发送一封图文+两个附件的邮件。要发送复杂邮件,得先熟悉三个概念,如下:
1.MimeMessage:该类是个能理解MIME类型和头的电子邮件消息
2.MimeMultipart:该类定义了增加、删除以及获取邮件不同部分内容的方法
3.MimeBodyPart:该对象代表一个MimeMessage对象内容的一部分。每个MimeBodyPart被认为有两部分:MIME类型和匹配这个类型的内容
完整的邮件生成过程如下(第一步和第三步参考上文):
MimeMessage mimeMessage = new MimeMessage(session); mimeMessage.addRecipients(Message.RecipientType.TO, "111@qq.com");//设置收信人 mimeMessage.addRecipients(Message.RecipientType.CC, "222@qq.com");//抄送 mimeMessage.setFrom("333@qq.com");//邮件发送人 mimeMessage.setSubject("测试邮件主题");//邮件主题 MimeMultipart mixed = new MimeMultipart("mixed"); mimeMessage.setContent(mixed);//设置整封邮件的MIME消息体为混合的组合关系 MimeBodyPart attach1 = new MimeBodyPart();//创建附件1 MimeBodyPart attach2 = new MimeBodyPart();//创建附件2 MimeBodyPart content = new MimeBodyPart();//创建邮件正文 mixed.addBodyPart(attach1);//将附件一添加到MIME消息体中 mixed.addBodyPart(attach2);//将附件二添加到MIME消息体中 mixed.addBodyPart(content);//将正文添加到消息体中 FileDataSource fds1 = new FileDataSource(new File("C://Users//sang//Desktop//1.png"));//构造附件一的数据源 DataHandler dh1 = new DataHandler(fds1);//数据处理 attach1.setDataHandler(dh1);//设置附件一的数据源 attach1.setFileName("1.png");//设置附件一的文件名 //附件二的操作与附件一类似,这里就不一一注释了 FileDataSource fds2 = new FileDataSource(new File("C://Users//sang//Desktop//博客笔记.xlsx")); DataHandler dh2 = new DataHandler(fds2); attach2.setDataHandler(dh2); attach2.setFileName(MimeUtility.encodeText("博客笔记.xlsx"));//设置文件名时,如果有中文,可以通过MimeUtility类中的encodeText方法进行编码,避免乱码 MimeMultipart bodyMimeMultipart = new MimeMultipart("related");//设置正文的MIME类型 content.setContent(bodyMimeMultipart);//将bodyMimeMultipart添加到正文消息体中 MimeBodyPart bodyPart = new MimeBodyPart();//正文的HTML部分 bodyPart.setContent("<h1>Hello大家好,这是一封测试邮件<img src='cid:2.png'/></h1>","text/html;charset=utf-8"); MimeBodyPart picPart = new MimeBodyPart();//正文的图片部分 DataHandler dataHandler = new DataHandler(new FileDataSource("C://Users//sang//Desktop//2.png")); picPart.setDataHandler(dataHandler); picPart.setContentID("2.png"); //将正文的HTML和图片部分分别添加到bodyMimeMultipart中 bodyMimeMultipart.addBodyPart(bodyPart); bodyMimeMultipart.addBodyPart(picPart); mimeMessage.saveChanges();
OK,Java Mail发送QQ邮件就是这么简单,至于其他的如163,sina等,写法类似,这里我就不赘述了。
小伙伴可能也注意到了,复杂邮件的构造其实有点麻烦,在项目中,我们可以使用Freemarker来构建邮件模板,这个模板问题我们下篇文章介绍。
本系列其他文章:
1. SpringBoot+Vue前后端分离,使用SpringSecurity完美处理权限问题(一)
2. SpringBoot+Vue前后端分离,使用SpringSecurity完美处理权限问题(二)
3. SpringSecurity中密码加盐与SpringBoot中异常统一处理
4. axios请求封装和异常统一处理
5. 权限管理模块中动态加载Vue组件
6. SpringBoot+Vue前后端分离,使用SpringSecurity完美处理权限问题(六)
7. vhr部门管理数据库设计与编程
8. 使用MyBatis轻松实现递归查询与存储过程调用
9. ElementUI中tree控件踩坑记
10. SpringBoot中自定义参数绑定
11. SpringBoot中使用POI,快速实现Excel导入导出
关注公众号,可以及时接收到最新文章: