SpringBoot快速实现发送邮件

1、添加发送邮件需要的maven依赖
在 pom.xml 配置文件中加入 spring-boot-starter-mail 依赖。

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

2、添加邮箱配置参数
如其他自动化配置模块一样,在完成了依赖引入之后,只需要在application.properties中配置相应的属性内容。

下面我们以QQ邮箱为例,注意替换成自己的发件邮箱和授权码

#邮箱配置
#平台地址,这里用的是qq邮箱,使用其他邮箱请更换
spring.mail.host=smtp.qq.com
#端口号
spring.mail.port=XXX
#发送邮件的邮箱地址:改成自己的邮箱
spring.mail.username=xxxxxxxxxxx
#发送短信后它给你的授权码 填写到这里
spring.mail.password=xxxxxxxxxxxx
#与发件邮箱一致
spring.mail.from=xxxxxxxxxxx

3、调用JavaMailSender接口发送邮件
由于Spring Boot的starter模块提供了自动化配置,所以在引入了spring-boot-starter-mail依赖之后,会根据配置文件中的内容去创建JavaMailSender实例,因此我们可以直接在需要使用的地方直接@Autowired来引入邮件发送对象。
(1)发送邮件工具类 MailUtil

package com.lcz.demo.mail;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.io.FileSystemResource;
import org.springframework.mail.SimpleMailMessage;
import org.springframework.mail.javamail.JavaMailSender;
import org.springframework.mail.javamail.MimeMessageHelper;
import org.springframework.stereotype.Service;
import javax.mail.MessagingException;
import javax.mail.internet.MimeMessage;
import java.io.File;/*** 发送邮件工具类 MailUtil** @author lcz* @date 2024/3/22 15:00*/@Service
public class EmailUtil implements EmailService
{private final Logger logger = LoggerFactory.getLogger(this.getClass());//Spring Boot 提供了一个发送邮件的简单抽象,使用的是下面这个接口,这里直接注入即可使用@Autowiredprivate JavaMailSender mailSender;// 配置文件中我的qq邮箱@Value("${spring.mail.from}")private String from;/*** 简单文本邮件* @param to 收件人* @param subject 主题* @param content 内容*/@Overridepublic void sendSimpleMail(String to, String subject, String content) {//创建SimpleMailMessage对象SimpleMailMessage message = new SimpleMailMessage();//邮件发送人message.setFrom(from);//邮件接收人message.setTo(to);//邮件主题message.setSubject(subject);//邮件内容message.setText(content);//发送邮件mailSender.send(message);}/*** html邮件* @param to 收件人,多个时参数形式 :"xxx@xxx.com,xxx@xxx.com,xxx@xxx.com"* @param subject 主题* @param content 内容*/@Overridepublic void sendHtmlMail(String to, String subject, String content) {//获取MimeMessage对象MimeMessage message = mailSender.createMimeMessage();MimeMessageHelper messageHelper;try {messageHelper = new MimeMessageHelper(message, true);//邮件发送人messageHelper.setFrom(from);//邮件接收人,设置多个收件人地址InternetAddress[] internetAddressTo = InternetAddress.parse(to);messageHelper.setTo(internetAddressTo);//messageHelper.setTo(to);//邮件主题message.setSubject(subject);//邮件内容,html格式messageHelper.setText(content, true);//发送mailSender.send(message);//日志信息logger.info("邮件已经发送。");} catch (Exception e) {logger.error("发送邮件时发生异常!", e);}}/*** 带附件的邮件* @param to 收件人* @param subject 主题* @param content 内容* @param filePath 附件*/@Overridepublic 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 (Exception e) {logger.error("发送邮件时发生异常!", e);}}
}

(2)EmailService接口

package com.lcz.demo.mail;/*** @author lcz*/public interface EmailService {/*** 发送文本邮件** @param to      收件人* @param subject 主题* @param content 内容*/void sendSimpleMail(String to, String subject, String content);/*** 发送HTML邮件** @param to      收件人* @param subject 主题* @param content 内容*/public void sendHtmlMail(String to, String subject, String content);/*** 发送带附件的邮件** @param to       收件人* @param subject  主题* @param content  内容* @param filePath 附件*/public void sendAttachmentsMail(String to, String subject, String content, String filePath);
}

(3)单元测试

package com.lcz.demo;import com.hs.demo.mail.EmailService;
import com.hs.demo.mail.EmailUtil;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;@RunWith(SpringRunner.class)
@SpringBootTest
public class SendEmailApplicationTests {/*** 注入发送邮件的接口*/@Autowiredprivate EmailUtil mailService;/*** 测试发送文本邮件*/@Testpublic void sendmail() {mailService.sendSimpleMail("xxx@qq.com","主题:你好普通邮件","内容:第一封邮件");}@Testpublic void sendmailHtml(){mailService.sendHtmlMail("xxx@qq.com","主题:你好html邮件","<h1>内容:第一封html邮件</h1>");}
}

4、常见的Web网站注册,邮箱点击链接验证激活如何实现?

功能:通过邮箱注册账号,注册成功会向邮箱发送激活邮件。提示用户登录邮箱进行账户激活,方可使用账号。
流程:本质上就是向user表里新增一条数据,user表中应有一个code字段存放随机串。code在添加用户时随机生成(uuid),发送邮件时把code值带到邮件链接中用于查找唯一账户,然后判断用户状态,进行激活。

5、存在失效邮箱地址导致JavaMailSender群发失败
问题描述

进行邮件群发,使用的是SpringBoot的org.springframework.mail.javamail.JavaMailSender。在提测过程中发现,如果待发送的邮件地址列表中存在一个无效的地址【该地址是一个合法的邮件地址,但是是无效地址,如:BingDwenDwen@163.com,它是一个合法的邮件地址,但却是无效的地址】,则会导致所有邮件发送失败。

报错异常和问题分析
异常:org.springframework.mail.MailSendException: Failed messages: javax.mail.SendFailedException: Invalid Addresses

分析:可能是收件人或抄送人列表存在无效的地址坑:不能直接catch到SendFailedException

解决方案

遍历异常,提取无效地址后过滤原地址列表再次发送
(1)发邮件方法代码

    /*** 发送html邮件** @param to* @param cc* @param subject* @param content*/public void sendHtmlMail(String[] to, String[] cc, String subject, String content) {MimeMessage message = mailSender.createMimeMessage();MimeMessageHelper helper = null;try {//true表示需要创建一个multipart messagehelper = new MimeMessageHelper(message, true);helper.setFrom(from);helper.setTo(to);helper.setCc(cc);helper.setSubject(subject);helper.setText(content, true);mailSender.send(message);logger.info("sendHtmlMail success.from:" + from);} catch (Throwable e) {logger.error("sendHtmlMail fail.", e);String[] invalid = getInvalidAddresses(e);if (invalid != null) {sendHtmlMail(filterByArray(to, invalid), filterByArray(cc, invalid), subject, content);}}}

(2)从异常获取无效地址的方法代码

    /*** 从异常获取无效地址* @param e* @return*/private static String[] getInvalidAddresses(Throwable e) {if (e == null) {return null;}if (e instanceof MailSendException) {System.out.println("e instanceof SendFailedException");Exception[] exceptions = ((MailSendException) e).getMessageExceptions();for (Exception exception : exceptions) {if (exception instanceof SendFailedException) {return getStringAddress(((SendFailedException) exception).getInvalidAddresses());}}}if (e instanceof SendFailedException) {return getStringAddress(((SendFailedException) e).getInvalidAddresses());}return null;}/*** 将Address[]转成String[]* @param address* @return*/private static String[] getStringAddress(Address[] address) {List<String> invalid = new ArrayList<>();for (Address a : address) {String aa = ((InternetAddress) a).getAddress();if (!StringUtils.isEmpty(aa)) {invalid.add(aa);}}return invalid.stream().distinct().toArray(String[]::new);}

(3)过滤发件人中无效地址的方法代码

    /*** 过滤数组source,规则为数组元素包含了数组filter中的元素则去除** @param source* @param filter* @return*/private static String[] filterByArray(String[] source, String[] filter) {List<String> result = new ArrayList<>();for (String s : source) {boolean contains = false;for (String f : filter) {if (s.contains(f)) {contains = true;break;}}if (!contains) {result.add(s);}}return result.stream().toArray(String[]::new);}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/787733.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

单源最短路径

题目描述 给定一个 n 个点&#xff0c;m 条有向边的带非负权图&#xff0c;请你计算从 s 出发&#xff0c;到每个点的距离。 数据保证你能从 s 出发到任意点。 输入格式 第一行为三个正整数n,m,s。 第二行起 m 行&#xff0c;每行三个非负整数 ui​,vi​,wi​&#xff0c;表…

关于v114之后的chromedriver及存放路径

使用selenium调用浏览器时&#xff0c;我一直调用谷歌浏览器&#xff0c;可浏览器升级后&#xff0c;就会再次遇到以前遇到过的各种问题&#xff0c;诸如&#xff1a;1、怎么关闭浏览器更新&#xff1b;2、去哪儿下载chromedriver&#xff1b;3、114版本之后的驱动去哪儿下载&a…

【御控物联】JavaScript JSON结构转换(11):数组To数组——综合应用

文章目录 一、JSON结构转换是什么&#xff1f;二、术语解释三、案例之《JSON数组 To JSON数组》四、代码实现五、在线转换工具六、技术资料 一、JSON结构转换是什么&#xff1f; JSON结构转换指的是将一个JSON对象或JSON数组按照一定规则进行重组、筛选、映射或转换&#xff0…

golang语言系列:golang基础知识

云原生学习路线导航页&#xff08;持续更新中&#xff09; 本文是 golang语言 系列文章&#xff0c;本节会对 golang 基础知识进行学习。gitee有个知识库https://gitee.com/yooome/golang&#xff0c;golang基础知识整理的非常详细了&#xff0c;本文直接链接到该git 0_Go语言的…

蓝桥杯真题:单词分析

import java.util.Scanner; //1:无需package //2: 类名必须Main, 不可修改 public class Main{public static void main(String[]args) {Scanner sannernew Scanner(System.in);String strsanner.nextLine();int []anew int [26];for(int i0;i<str.length();i) {a[str.charA…

记录几个常用命令

目录 一、查询历史命令 二、启动java容器 三、启动java容器并联通mysql容器 一、查询历史命令 # 查出所有"docker run"的历史命令 history | grep "docker run" 二、启动java容器 docker run -itd --name atcc-door -v /home/wwwroot/atcc-door:/hom…

基于蚁群算法的三维路径规划(matlab实现)

作品简介 1 理论基础 1.1 三维路径规划问题概述 三维路径规划指在已知三维地图中&#xff0c;规划出一条从出发点到目标点满足某项指标最优&#xff0c;并且避开了所有三维障碍物的三维最优路径。现有的路径规划算法中&#xff0c;大部分算法是在二维规划平面或准二维规划平面…

分享一种快速移植OpenHarmony Linux内核的方法

移植概述 本文面向希望将 OpenHarmony 移植到三方芯片平台硬件的开发者&#xff0c;介绍一种借助三方芯片平台自带 Linux 内核的现有能力&#xff0c;快速移植 OpenHarmony 到三方芯片平台的方法。 移植到三方芯片平台的整体思路 内核态层和用户态层 为了更好的解释整个内核…

HP Z440不重启进入bios,在服务器系统里修改bios的配置

HP Z440不重启进入bios&#xff0c;在服务器系统里修改bios的配置 在某些情况下&#xff0c;你可以在不重启并进入BIOS/UEFI设置界面的情况下&#xff0c;直接从操作系统内部修改BIOS配置。这通常通过特定的软件工具实现&#xff0c;这些工具能够与系统的固件层进行交云&#…

校园跑腿的含义是什么?大学里校园跑腿的特点有哪些?

校园跑腿是指校园内的一种学生间互助服务活动&#xff0c;即学生通过平台发布需求&#xff0c;由其他学生以跑腿的方式提供相应服务&#xff0c;以获取服务费。这种服务模式为需求者提供便利&#xff0c;同时也为提供服务的学生带来一定的收入。 大学里校园跑腿的特点主要有以…

git实战教程

Git实战教程涵盖了Git的基本操作以及如何在日常开发中高效地使用Git进行版本控制。以下是一些Git实战的基础操作步骤&#xff0c;以帮助您快速入门和熟悉Git的使用&#xff1a; ### Git基础操作流程 1. **安装Git** - 下载并安装适用于您操作系统的Git客户端&#xff0c;比…

若依分离版 —引入echart连接Springboot后端

1. vue引入echart &#xff08;1&#xff09;首先安装ECharts库。可以通过npm npm install echarts --save &#xff08;2&#xff09;在vue页面中添加一个容器元素来显示图表 <el-card class"mt20"><div id"ha" ref"main"><…

六、保持长期高效的七个法则(二)Rules for Staying Productive Long-Term(2)

Rule #5 - If your work changes, your system should too. 准则五&#xff1a;如果你的工作变了&#xff0c;你的系统也应该改变。 For some, work will be consistent enough to not need major changes.You simply stick to the same system and you’ll get the results y…

fbx smpl转换

目录 3.7 3.8版本&#xff1a; c sdk 3.9 自己编译 下面的库和代码结合 测试ok 大神编译了各个版本 smpl转fbx windows 可行代码 3.7 3.8版本&#xff1a; https://github.com/Shiiho11/FBX-Python-SDK-for-Python3.x c sdk vs2019 https://www.autodesk.com/develop…

openGauss 分区

分区 可获得性 本特性自openGauss 1.1.0版本开始引入。 特性简介 数据分区是在一个节点内部对数据按照用户指定的策略做进一步的水平分表&#xff0c;将表按照指定范围划分为多个数据互不重叠的部分。 客户价值 对于大多数用户使用场景&#xff0c;分区表和普通表相比具有…

【Spring】SpringBoot整合MybatisPlus的基本应用

&#x1f4dd;个人主页&#xff1a;哈__ 期待您的关注 一、MybatisPlus简介 先来看一下官方的简介吧。 MyBatis-Plus &#xff08;简称 MP&#xff09;是一个 MyBatis的增强工具&#xff0c;在 MyBatis 的基础上只做增强不做改变&#xff0c;为 简化开发、提高效率而生。Myb…

安卓Android 架构模式及UI布局设计

文章目录 一、Android UI 简介1.1 在手机UI设计中&#xff0c;坚持的原则是什么1.2 安卓中的架构模式1.2.1 MVC (Model-View-Controller)设计模式优缺点 1.2.2 MVP(Model-View-Presenter)设计模式MVP与MVC关系&#xff1a; 1.2.3 MVVM(Model—View—ViewModel ) 设计模式1.2.4 …

关于 C/C++ 1Z(17)开源项目 openppp2 协同程式切换工作流

下述为开源项目 openppp2&#xff08;github&#xff09;构建工作在 C/C 17 的 stackful 有栈协同程式的工作流切换示意图&#xff1a; 在 openppp2 之中采用人工手动方式管理协同程式之间的切换&#xff0c;每个中断过程只是保存线程栈信息&#xff08;如寄存器、当前#PC EIP&…

Python PyQt5——QPainter 绘图用法与代码示例

QPainter是Qt框架中的一个类&#xff0c;用于在Qt基于应用程序中进行绘图和绘制图形。 使用QPainter&#xff0c;可以绘制各种形状&#xff0c;如线条、矩形、椭圆和多边形&#xff0c;以及文本和图像。它提供了用于指定颜色、画笔、笔刷和变换的方法。 QPainter 在 PyQt5 中…

Deferred library xxx was not loaded

Deferred 延迟修饰词作用下的文件库尚未完成载入