天天看點

(轉)Springboot郵件服務

springboot仍然在狂速發展,才五個多月沒有關注,現在看​​官網​​已經到1.5.3.RELEASE版本了。準備慢慢在寫寫springboot相關的文章,本篇文章使用springboot最新版本1.5.3進行開發。

發送郵件應該是網站的必備功能之一,什麼注冊驗證,忘記密碼或者是給使用者發送營銷資訊。最早期的時候我們會使用JavaMail相關api來寫發送郵件的相關代碼,後來spring推出了JavaMailSender更加簡化了郵件發送的過程,在之後springboot對此進行了封裝就有了現在的spring-boot-starter-mail,本章文章的介紹主要來自于此包。

簡單使用

1、pom包配置

pom包裡面添加spring-boot-starter-mail包引用

<dependencies>
  <dependency> 
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-mail</artifactId>
  </dependency> 
</dependencies>      

2、在application.properties中添加郵箱配置

spring.mail.host=smtp.qiye.163.com //郵箱伺服器位址
[email protected] //使用者名
spring.mail.password=xxyyooo    //密碼,(qq,163是用戶端授權碼)
spring.mail.default-encoding=UTF-8

[email protected]  //以誰來發送郵件      

3、編寫mailService,這裡隻提出實作類。

@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);
        }

    }
}      

4、編寫test類進行測試

@RunWith(SpringRunner.class)
@SpringBootTest
public class MailServiceTest {

    @Autowired
    private MailService MailService;

    @Test
    public void testSimpleMail() throws Exception {
        MailService.sendSimpleMail("[email protected]","test simple mail"," hello this is simple mail");
    }
}      

至此一個簡單的文本發送就完成了。

加點料

但是在正常使用的過程中,我們通常在郵件中加入圖檔或者附件來豐富郵件的内容,下面講介紹如何使用springboot來發送豐富的郵件。

發送html格式郵件

其它都不變在MailService添加sendHtmlMail方法.

public void sendHtmlMail(String to, String subject, String content) {
    MimeMessage message = mailSender.createMimeMessage();

    try {
        //true表示需要建立一個multipart message
        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内容,測試發送

@Test
public void testHtmlMail() throws Exception {
    String content="<html>\n" +
            "<body>\n" +
            "    <h3>hello world ! 這是一封Html郵件!</h3>\n" +
            "</body>\n" +
            "</html>";
    MailService.sendHtmlMail("[email protected]","test simple mail",content);
}      

發送帶附件的郵件

在MailService添加sendAttachmentsMail方法.

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)​

在測試類中添加測試方法

@Test
public void sendAttachmentsMail() {
    String filePath="e:\\tmp\\application.log";
    mailService.sendAttachmentsMail("[email protected]", "主題:帶附件的郵件", "有附件,請查收!", filePath);
}      

發送帶靜态資源的郵件

郵件中的靜态資源一般就是指圖檔,在MailService添加sendAttachmentsMail方法.

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);
    }
}      
@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("[email protected]", "主題:這是有圖檔的郵件", content, imgPath, rscId);
}      
添加多個圖檔可以使用多條 ​

​<img src='cid:" + rscId + "' >​

​ 和 ​

​helper.addInline(rscId, res)​

​ 來實作

到此所有的郵件發送服務已經完成了。

郵件系統

上面發送郵件的基礎服務就這些了,但是如果我們要做成一個郵件系統的話還需要考慮以下幾個問題:

郵件模闆

我們會經常收到這樣的郵件:

尊敬的neo使用者:

              恭喜您注冊成為xxx網的使用者,,同時感謝您對xxx的關注與支援并歡迎您使用xx的産品與服務。
              ...

      

其中隻有neo這個使用者名在變化,其它郵件内容均不變,如果每次發送郵件都需要手動拼接的話會不夠優雅,并且每次模闆的修改都需要改動代碼的話也很不友善,是以對于這類郵件需求,都建議做成郵件模闆來處理。模闆的本質很簡單,就是在模闆中替換變化的參數,轉換為html字元串即可,這裡以​

​thymeleaf​

​為例來示範。

1、pom中導入thymeleaf的包

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>      

2、在resorces/templates下建立emailTemplate.html

<!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.ityouknow.com/neo/{id}(id=${id}) }">激活賬号</a>
    </body>
</html>      

3、解析模闆并發送

@Test
public void sendTemplateMail() {
    //建立郵件正文
    Context context = new Context();
    context.setVariable("id", "006");
    String emailContent = templateEngine.process("emailTemplate", context);

    mailService.sendHtmlMail("[email protected]","主題:這是模闆郵件",emailContent);
}      

發送失敗

因為各種原因,總會有郵件發送失敗的情況,比如:郵件發送過于頻繁、網絡異常等。在出現這種情況的時候,我們一般會考慮重新重試發送郵件,會分為以下幾個步驟來實作:

  • 1、接收到發送郵件請求,首先記錄請求并且入庫。
  • 2、調用郵件發送接口發送郵件,并且将發送結果記錄入庫。
  • 3、啟動定時系統掃描時間段内,未發送成功并且重試次數小于3次的郵件,進行再次發送

異步發送

很多時候郵件發送并不是我們主業務必須關注的結果,比如通知類、提醒類的業務可以允許延時或者失敗。這個時候可以采用異步的方式來發送郵件,加快主交易執行速度,在實際項目中可以采用MQ發送郵件相關參數,監聽到消息隊列之後啟動發送郵件。​​​​

繼續閱讀