Java 发送邮件,字节跳动上千道精选面试题还不刷起来
@GetMapping("/send")
public String sendEmil() throws Exception {
javaMailUtils.send("javaMail 邮件主题","zks_2019@163.com",
"这里是邮件的主体内容");
return "success";
}
}
邮件展示
发送带图片,带附件的邮件(此处的图片、附件都为本地图片,下面会展示如何发送网络附件)
public void sendAttachFileMail(String subject,String to,String text) throws MessagingException {
MimeMessage mimeMessage = javaMailSender.createMimeMessage();
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage,true);
helper.setFrom(username);
helper.setTo(to);
helper.setSubject(subject);
helper.setSentDate(new Date());
helper.setText("<p>"+text+"</p>"+"<img src = 'cid:p01' /",true);
helper.addInline("p01",
new FileSystemResource(new File("C:\Users\admin\Desktop\zhoukaishun\kaishun.jpg")));
helper.addAttachment("文件.docx",new File("C:\Users\admin\Desktop\测试邮件附件信息.docx"));
javaMailSender.send(mimeMessage);
}
controller 调用
@RestController
@RequestMapping("emil")
public class EmilController {
@Resource
private JavaMailUtils javaMailUtils;
@GetMapping("/send")
public String sendEmil() throws Exception {
javaMailUtils.sendAttachFileMail("带有图片、附件的邮件主题","zks_2019@163.com",
"这是带有图片和附件的一份邮件,并且支持 html 标签<h1>这是一个 h1 标签</h1>");
return "success";
}
}
邮件展示
这里既可以满足日常开发的一般需求。但是还有一种可能性,如果我的项目属于 sass 服务,有很多的租户,每个租户的发件人肯定是不一样的,我不能都去改配置文件吧,所以我们还可以通过不需要配置 yml,也不通过 javaMailSender 去调用,采用手动配置的方式去调用。这样后续可扩展性比较高,比如我们设计一个邮件配置表,里面绑定租户与发件人,每次通过租户的信息去读表选择谁来发送
还以 QQ 邮箱发送为例
二、手动配置发送
public void send() throws Exception {
Properties props = new Properties();
// 开启 debug 调试
props.setProperty("mail.debug", "true");
// 发送服务器需要身份验证
props.setProperty("mail.smtp.auth", "true");
// 设置邮件服务器主机名
props.setProperty("mail.host", "smtp.qq.com");
// 发送邮件协议名称
props.setProperty("mail.transport.protocol", "smtp");
// 配置 ssl 加密工厂
MailSSLSocketFactory sf = new MailSSLSocketFactory();
sf.setTrustAllHosts(true);
props.put("mail.smtp.ssl.enable", "true");
props.put("mail.smtp.ssl.socketFactory", sf);
Session session = Session.getInstance(props);
//构建邮件详情
MimeMessage mimeMessage = createMimeMessage(session);
session.setDebug(false);
Transport transport = session.getTransport();
transport.connect("smtp.qq.com", "发件人邮箱", "发件人的授权码");
transport.sendMessage(mimeMessage, new Address[] { new InternetAddress("zks_2019@163.com") });
trans
port.close();
}
private MimeMessage createMimeMessage(Session session) throws Exception{
// 1. 创建邮件对象
MimeMessage mimeMessage = new MimeMessage(session);
// 2. From: 发件人
mimeMessage.setFrom(new InternetAddress("发件人邮箱", "zhoukaishun", "UTF-8"));
// 3. To: 收件人(可以增加多个收件人、抄送、密送)
mimeMessage.addRecipient(MimeMessage.RecipientType.TO, new InternetAddress("zks_2019@163.com", "kaishun",
"UTF-8"));
// 4. Subject: 邮件主题
mimeMessage.setSubject("测试邮件", "UTF-8");
// 5. 创建图片“节点”
MimeBodyPart image = new MimeBodyPart();
DataHandler dataHandlers = new DataHandler(new URL("http://www.zhoukaishun.com:8001/static/kaishun.jpg"));
image.setDataHandler(dataHandlers);
// 读取本地文件
//DataHandler dhImage = new DataHandler(new FileDataSource("C:\Users\admin\Desktop\logo.png"));
// 将图片数据添加到“节点”
//image.setDataHandler(dhImage);
// 为“节点”设置一个唯一编号(在文本“节点”将引用该 ID)
image.setContentID("image_fairy_tail");
// 6. 创建文本“节点”
MimeBodyPart text = new MimeBodyPart();
// 这里添加图片的方式是将整个图片包含到邮件内容中, 实际上也可以以 http 链接的形式添加网络图片
text.setContent("<p>邮件内容</p>"+"<img src='cid:image_fairy_tail'/>", "text/html;charset=UTF-8");
// 7. (文本+图片)设置 文本 和 图片 “节点”的关系(将 文本 和 图片 “节点”合成一个混合“节点”)
MimeMultipart mmTextImage = new MimeMultipart();
mmTextImage.addBodyPart(text);
mmTextImage.addBodyPart(image);
// 关联关系
mmTextImage.setSubType("related");
// 8. 将 文本+图片 的混合“节点”封装成一个普通“节点”
// 最终添加到邮件的 Content 是由多个 BodyPart 组成的 Multipart, 所以我们需要的是 BodyPart,
// 上面的 mmTextImage 并非 BodyPart, 所有要把 mmTextImage 封装成一个 BodyPart
MimeBodyPart textImage = new MimeBodyPart();
textImage.setContent(mmTextImage);
// 9. 创建附件“节点”
MimeBodyPart attachment = new MimeBodyPart();
// 读取本地文件
DataHandler dhFile = new DataHandler(new FileDataSource("C:\Users\admin\Desktop\测试邮件附件信息.docx"));
// 将附件数据添加到“节点”
attachment.setDataHandler(dhFile);
// 设置附件的文件名(这里有个坑点,之前看博客说需要编码,但是如果文件名过长会导致乱码,且点进 attachment.setFileName 可以发现,底层已经帮你编码了。所以这里不需要编码。如果文件名不长不会出现问题,如果文件名称过长,则会体现为乱码)
attachment.setFileName(MimeUtility.encodeText(dhFile.getName()));
//邮件中包含网络附件
String path2 = "http://www.zhoukaishun.com:8001/static/JVM.docx";
URL url = new URL(path2);
DataHandler dataHandler = new DataHandler(url);
MimeBodyPart messageBodyPart3 = new MimeBodyPart();
messageBodyPart3.setDataHandler(dataHandler);
messageBodyPart3.setFileName("jvm.docx");
// 10. 设置(文本+图片)和 附件(若有) 的关系(合成一个大的混合“节点” / Multipart )
MimeMultipart mm = new MimeMultipart();
mm.addBodyPart(textImage);
mm.addBodyPart(attachment);
mm.addBodyPart(messageBodyPart3);
// 混合关系
mm.setSubType("mixed");
// 11. 设置整个邮件的关系(将最终的混合“节点”作为邮件的内容添加到邮件对象)
mimeMessage.setContent(mm);
// 12. 设置发件时间
mimeMessage.setSentDate(new Date());
// 13. 保存上面的所有设置
mimeMessage.saveChanges();
return mimeMessage;
}
controller 调用
@RestController
@RequestMapping("emil")
public class EmilController {
@Resource
private JavaMailUtils javaMailUtils;
@GetMapping("/send")
public String sendEmil() throws Exception {
javaMailUtils.send();
return "success";
}
}
评论