发送邮件的JavaMail和Spring提供的MailSender,以及比较


发邮件,项目的必备功能之一,如果一个稍微模块化一点的公司,一般会单独出来一个项目专用来做公司的发送信息的功能,当然这个发送信息中不止包含发邮件,还会有短信、APP push等。这篇聊聊推送邮件。

在以前的开发中,公司用Java mail的比较多,由自己来写邮件的组装和发送功能,但是Java mail使用操作比较繁杂,后来渐渐的都开始使用spring提供的JavaMailSender工具来实现,用过的都知道,这叫一个爽,执行几个set、add操作,一个复杂的邮件就可以发送出去,但是在业务代码中发送邮件的位置很多,不能在每个业务逻辑位置都写一遍同样的set操作,这是基本常识,因此都会基于JavaMailSender再做一个工具类,这个工具类让调用者只用一句话实现就OK。引出主题工具类,接下来就开干吧。

1. JavaMail邮件推送

这边的主要内容是写JavaMailSender,但是开始还是来看看JavaMail的基本实现,因为JavaMailSender也是基于JavaMail做的再次封装。

1.1 邮件主要组成部分

发送邮件的三个重点:

  • Session:是用来设置邮件参数以及和邮件服务器建立会话用的。
  • MimeMessage:邮件本体,所有的邮件内容、邮件的收件人、发送人、附件、内嵌图片等都是在这里面封装。
  • Transport:提供发送邮件的API。

Session中实际也提供了获取transport的方法,但是这个transport不好用,还是习惯用Transport提供的静态调用。

/*Transport静态方法直接调用,将message传入即可*/
Transport.send(message);
/*Session获取transport方式调用*/
Transport transport = session.getTransport();
//address需要手动添加,表示发送人
transport.sendMessage(message,new Address[1]);

上面Session的transport只能指定发送人,抄送人、密送人都不能添加,如果在MimeMessage中添加不知道能不能起作用,这个没有尝试过,如果你喜欢用这种方式可以分享一下。

1.2 引入依赖

Java mail有两个依赖,分别是mail和activation。

<dependency>
    <groupId>javax.mail</groupId>
    <artifactId>mail</artifactId>
    <version>1.4.7</version>
</dependency>
<dependency>
    <groupId>javax.activation</groupId>
    <artifactId>activation</artifactId>
    <version>1.1.1</version>
</dependency>

1.3 简单邮件

简单的发送邮件没有复杂的逻辑,代码也很清晰,直接上代码:

private static void send() throws MessagingException {
    //构建Session
    Properties properties = new Properties();
    properties.setProperty("mail.host","smtp.163.com");
    properties.setProperty("mail.transport.protocol","smtp");
    properties.setProperty("mail.smtp.auth","true");
    properties.setProperty("mail.host.port","25");
    Session session = Session.getDefaultInstance(properties, new Authenticator() {
        @Override
        protected PasswordAuthentication getPasswordAuthentication() {
            return new PasswordAuthentication("xxx@163.com","xxxxxx");
        }
    });
    
    session.setDebug(true);
    //构建邮件MimeMessage
    MimeMessage message = new MimeMessage(session);
    message.setSubject("这是一封测试邮件!");
    message.setText("测试邮件内容随意");
    message.setRecipient(MimeMessage.RecipientType.TO,new InternetAddress("itcrud@aliyun.com"));
    message.setFrom(new InternetAddress("xxx@163.com"));

    //发送邮件
    Transport.send(message);
}

需要注意的点:

  • 发件人要和进行密码验证的邮箱保持一致,否则会报错:com.sun.mail.smtp.SMTPSendFailedException: 553 Mail from must equal authorized user
  • 如果要发抄送人和密送人,需要使用setRecipient方法,在第一参数中指定抄送类型即可。

1.4 复杂邮件

复杂邮件使用的相对会比较多一点,在发送邮件的时候多少会带上内嵌图片或者附件。MimeMessage是整个邮件本体,构建复杂邮件的过程就是向这个本体中加不同的内容,Java mail提供了一个类MimeBodyPart,表示邮件内容的组件。将需要发送的内容用组件包装起来,再塞到邮件本地中即可。

示例代码:

private static void send() throws MessagingException {
    //构建Session
    Properties properties = new Properties();
    properties.setProperty("mail.host","smtp.163.com");
    properties.setProperty("mail.transport.protocol","smtp");
    properties.setProperty("mail.smtp.auth","true");
    properties.setProperty("mail.host.port","25");
    Session session = Session.getDefaultInstance(properties, new Authenticator() {
        @Override
        protected PasswordAuthentication getPasswordAuthentication() {
            return new PasswordAuthentication("xxx@163.com","xxxxx");
        }
    });
    session.setDebug(true);//打印发送日志

    //构建基础信息
    MimeMessage message = new MimeMessage(session);
    message.setSubject("这是一封测试邮件!");
    //message.setText("测试邮件内容随意");
    message.setRecipient(MimeMessage.RecipientType.TO,new InternetAddress("itcrud@aliyun.com"));
    message.setFrom(new InternetAddress("xxx@163.com"));

    //构建内容
    MimeBodyPart text = new MimeBodyPart();
    text.setContent("<html><body><h3>你好,这是一封模板邮件!</h3><img src='cid:avatar'><h3>上面内嵌了一张图片↑↑↑↑</h3></body></html>", "text/html;charset=UTF-8");
    MimeBodyPart bodyPart = new MimeBodyPart();

    //构建内嵌图片
    DataHandler dataHandler = new DataHandler(new FileDataSource("/Users/joker/Downloads/avatar.png"));
    bodyPart.setDataHandler(dataHandler);
    bodyPart.setContentID("avatar");//如果没有这句,表示作为附件发送,加上表示显示在邮件内容的一部分

    //构建附件
    MimeBodyPart attachment = new MimeBodyPart();
    DataHandler attachmentHandler = new DataHandler(new FileDataSource("/Users/joker/Downloads/avatar.png"));
    attachment.setDataHandler(attachmentHandler);
    attachment.addHeader("Content-Type","UTF-8");//防止中文名乱码
    attachment.setFileName("avatar.png");//这里名称记得带上后缀名,否则邮件附件下载下来需要手动添加后缀,可能会损坏

    //构建邮件体内容
    MimeMultipart mimeMultipart = new MimeMultipart();
    mimeMultipart.addBodyPart(text);
    mimeMultipart.addBodyPart(bodyPart);
    mimeMultipart.addBodyPart(attachment);
    mimeMultipart.setSubType("mixed");//设置为mixed,别忘了哦
    message.setContent(mimeMultipart);

    //发送邮件
    Transport.send(message);
}

复杂邮件构建的过程不是很难,就是代码量会多一点。

2. JavaMailSender

JavaMailSender是由spring提供的,功能很强大,对Java mail做了一层封装,在Spring boot中可以拿来用,是自动集成的。

2.1 引入依赖

和Spring boot整合后,引入依赖就很简单啦。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-mail</artifactId>
</dependency>
<!--是在使用邮件模板的时候使用,不使用模板可以忽略-->
<dependency>
    <groupId>org.apache.velocity</groupId>
    <artifactId>velocity</artifactId>
    <version>1.6.4</version>
</dependency>

2.2 配置信息

使用JavaMailSender需要在配置文件中配置相关的信息,在Java mail中是通过Session来配置,在JavaMailSender上,在Spring项目中可以通过在配置文件中配置。(如果使用的是spring boot,在properties中配置,如果是Spring MVC项目就要用xml文件啦,或者自己手动的使用Configuration类来配置,本文以Spring boot为基础)

spring.mail.default-encoding=utf-8
spring.mail.host=smtp.163.com
spring.mail.protocol=smtp
spring.mail.properties.mail.smtp.auth=true
spring.mail.properties.mail.transport.protocol=smtp
spring.mail.properties.mail.smtp.starttls.enable=true
spring.mail.properties.mail.smtp.starttls.required=true
spring.mail.username=xxx@163.com
spring.mail.password=xxxx

2.3 简单邮件

简单邮件发送方法,到底简单到什么程度,看下面的代码:

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = ItcrudCommonsUtilApplication.class)
public class JavaMailSenderTest {

    @Autowired
    private JavaMailSender javaMailSender;

    @Autowired
    private VelocityEngine velocityEngine;//暂时用不上,在下面说到模板邮件使用

    @Test
    public void sendTest() throws MessagingException {
        //创建邮件本体
        MimeMessage message = javaMailSender.createMimeMessage();
        //创建本体编辑的协助者(YY出来的名字)
        MimeMessageHelper helper = new MimeMessageHelper(message);
		//构建邮件本体
        helper.setSubject("邮件标题");
        helper.setText("发送的内容");
        helper.setTo("itcrud@aliyun.com");
        helper.setFrom("xxx@163.com");
		//发送邮件
        javaMailSender.send(helper.getMimeMessage());
    }
}

看了这段发送邮件的代码,然后和上面Java mail的代码对比,果断的舍弃Java mail的有木有。

2.4 复杂邮件

对于Java mail来说是复杂邮件,但是对于JavaMailSender来说,复杂也会变得简单。

示例代码:

@Test
public void sendComplexText() throws MessagingException {
    MimeMessage message = javaMailSender.createMimeMessage();
    MimeMessageHelper helper = new MimeMessageHelper(message, true);
    //构建邮件本体内容
    helper.setSubject("复杂邮件");
    helper.setFrom("xxx@163.com");
    helper.setTo("itcrud@aliyun.com");
    helper.setText("<html><body><h3>你好,这是一封模板邮件!</h3><img src='cid:avatar'><h3>上面内嵌了一张图片↑↑↑↑</h3></body></html>", true);
    helper.addInline("avatar", new FileDataSource("/Users/joker/Downloads/avatar.png"));
    helper.addAttachment("avatar.png",new FileDataSource("/Users/joker/Downloads/avatar.png"));
    //发送邮件
    javaMailSender.send(helper.getMimeMessage());
}

代码的减少量可以看得出来吧。那些复杂的封装逻辑就交给Spring来处理啦。

需要注意的两点:(刚开始用的时候被坑过)

  • 在创建MimeMessageHelper的时候要指定是否为multipart(构造方法的第二个参数),可以理解为复杂邮件的意思;
  • 在setText方法的第二个参数也是要指定为true,如果这里没有设置true,这个内容会直接显示在邮件中,不会解析html内容。

3. 模板邮件

上面的两个内容都不是最主要的,在将Java mail的时候最后一个功能很难说,代码量太多。下面我们用JavaMailSender来实现。

场景:在公司对外发送邮件的时候,除了附件、内嵌图片和普通文本,还有两个很重的东西,那就是签名和占位填充。个人发邮件的时候都会带上自己的签名,何况是公司向外发邮件啦。另外一个邮件的基本格式相同,只是部分填充内容不同,总不能每次发送文件都要重新写一个html文件吧。

但是想想,邮件签名就直接加在内容里面不就可以啦,然后使用html格式来进行邮件的排版。对的没有错,但是如果整个邮件的排版比较复杂,html内容很长怎么办。那也没问题啊,可以放在文件里面,通过流来读取文件不就可以了吗。对的的确是这样的,但是如果使用Java mail就要去手动的读取问题,不觉得会很麻烦吗。还有就是占位符的填充。

这个时候JavaMailSender帮我们解决了这一系列的问题。准确的说是和另一个插件Velocity结合来解决。这就是在引入依赖的时候,我加上velocity的依赖的重点所在。

只需要两步就可以搞定:

  • 在resources目录下建立一个目录用来存在模板文件。(这里指定mail目录)
<!--模板文件的内容-->
<html>
    <body>
        <h1>这是邮件的标题</h1>
        <div>
            <p>很长很长的邮件内容上半部分</p>
            <p>很长很长的邮件内容上半部分</p>
        </div>
        <img src="cid:avatar">
        <div>
            <p>很长很长的邮件内容下半部分</p>
            <p>很长很长的邮件内容下半部分</p>
            <p>还有一部分内容看附件</p>
        </div>
        <div>
            <p>
                这里是签名的内容
                公司名称:xxxx有限责任公司
                联系电话:xxxxxxxx
            </p>
        </div>
    </body>
</html>
  • 在代码中使用VelocityEngine来读取文件,并发送邮件
@Test
public void sendSuperTest() throws MessagingException {
    MimeMessage message = javaMailSender.createMimeMessage();
    MimeMessageHelper helper = new MimeMessageHelper(message, true);
    //构建邮件本体内容
    helper.setSubject("复杂邮件");
    helper.setFrom("xxx@163.com");
    helper.setTo("itcrud@aliyun.com");
    //使用
    VelocityContext context = new VelocityContext();
    context.put("content","占位符需要填充的内容");
    try (StringWriter writer = new StringWriter()) {
        velocityEngine.mergeTemplate("/mail/temp.vm", "UTF-8", context, writer);
        helper.setText(writer.toString(), true);
    } catch (Exception e) {
        e.printStackTrace();
    }
    helper.addInline("avatar", new FileDataSource("/Users/joker/Downloads/avatar.png"));
    helper.addAttachment("avatar.png",new FileDataSource("/Users/joker/Downloads/avatar.png"));
    //发送邮件
    javaMailSender.send(helper.getMimeMessage());
}

完美解决上面的两个问题,并且整个实现过程的代码量也是很少的,比上面Java mail发送一个复杂邮件还要少。

4. 总结

这里讲到了Java mail发送邮件和Spring封装后的JavaMailSender,两者对比来说,JavaMailSender要比Java mail直接实现要简单的多。如果你有时间其实可以自己实现一套Spring的这种封装,但是为何要重复造轮子呢。

5. 源代码

GitHub:https://github.com/itcrud/itcrud-commons(mail)


文章作者: 程序猿洞晓
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 程序猿洞晓 !
评论
 上一篇
Spring Data JPA使用必备(一):Spring Data JPA整合Spring以及简单的使用 Spring Data JPA使用必备(一):Spring Data JPA整合Spring以及简单的使用
最近公司部分新开的项目需要使用Spring data JPA技术,作为一个从来没有用过JPA的小白来说,需要重新的学习。N年前简单的看过JPA,这么多年没用过,完全忘记了有木有。接下来的系列文章就一起来整理一下。使用了@Entity注解后会默认映射到数据库的表,按照驼峰规则,如类名是CustomerBaseInfo,那么就会对应到表名为……
下一篇 
Java输入和输出流关闭的顺序和关闭的姿势对比理解 Java输入和输出流关闭的顺序和关闭的姿势对比理解
Java的流操作在实际应用中使用的很多,但是流的关闭顺序到底有没有要求,关闭流的顺序和节点流与处理流有什么关系,输出流和输入流又有什么区别,然后就是关闭流可以通过哪几种方式。这篇文章将会和大家讨论一下。……。整个篇幅说了节点流和处理流的区别,然后根据输出流和输入流讨论流的关闭顺序问题,然后就是流不同的关闭姿势说明,最后比较流不同关闭姿势的优缺点。
2018-10-25
  目录