一、JavaWeb 文件上传
1.1 概述
-
配置 Servlet 和 jsp 环境;
-
文件上传需要的 jar 包:
-
或导入 Maven 依赖:
<dependency> <groupId>commons-io</groupId> <artifactId>commons-io</artifactId> <version>2.11.0</version> </dependency> <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.4</version> </dependency>
-
文件上传,注意事项:
- 为保证服务器安全,上传文件应放在外界无法直接访问的目录下,如:
WEB-INF
目录下; - 为防止文件被覆盖,要为上传文件,生成一个 唯一的文件名;
- 要 限制 上传文件的 最大值;
- 可以限制上传文件的类型,在收到上传文件名时,判断后缀名是否合法;
- 表单 method 属性,应该设置为 POST 方法,不能使用 GET 方法;
- 为保证服务器安全,上传文件应放在外界无法直接访问的目录下,如:
1.2 使用到的类
- ServletFileUpload
- 负责处理上传的文件数据,并将表单中每个输入项,封装成一个
FileItem
对象;-
parseRequest(HttpServletRequest)
方法,可以将表单中,每一个 HTML 标签提交的数据,封装成一个FileItem
对象,然后以 List 列表的形式返回;
-
- 在使用
ServletFileUpload
对象解析请求时,需要用到DiskFileItemFactory
对象,所以,使用前,需要先构造好DiskFileItemFactory
对象; - 通过
ServletFileUpload
对象的构造方法,或setFileItemFactory()
方法,设置ServletFileUpload
对象的fileItemFactory
属性。
- FileItem
-
必须设置
input
输入项的name
属性为file
,否则,浏览器不会发送上传文件的数据;-
<input type="file" name="filename">
;
-
-
表单如果包含一个文件上传输入项的话,这个表单的
enctype
属性就必须设置为multipart/form-data
; -
表单:
<%-- 通过表单上传文件 get:上传文件大小由限制 post:大小没限制 --%> <form action="" enctype="multipart/form-data" method="post"> <p><input type="file" name="fileName"></p> <p> <input type="submit" name="submit"> <input type="reset" name="reset"> </p> </form>
-
浏览器表单的类型,如果为
multipart/form-data
,在服务器端要获取数据,就要通过流。 -
常用方法:
方法名 | 描述 |
---|---|
boolean isFormField() | 判断是否是文本表单,是返回 true,否则返回 false |
string getFieldName() | 返回表单标签 name 的属性值 |
string getstring() | 数据流内容,以字符串返回 |
string getName() | 获得上传文件名 |
Inputstream getInputstream() | 以流的形式,返回上传文件的数据内容 |
void delete() | 删除文件 |
- DiskFileItemFactory
- 将请求消息实体中的每一个项目,封装成单独的
DiskFileItem
; -
FileItem
接口的实现对象,由FileItemFactory
接口,默认实现DiskFileItemFactory
; - 当上传的文件较小时,直接保存在内存中(速度快),文件较大时,以临时文件的形式,保存在磁盘临时文件夹(速度慢,但是内存资源是有限的)。
1.3 代码的实现逻辑
- 判断用户上传的文件,是普通表单,还是带文件的表单,如果是普通文件,直接返回(return);
- 创建文件上传保存的路径,
WEB-INF
路径下是安全的,用户无法直接访问; - 创建临时文件路径,如果文件超过预期大小,放到一个临时文件中;
- 创建
DiskFileItemFactory
对象,设置文件临时路径setRepository
,或大小的限制(设置缓冲区,setSizeThreshold
); - 获取
ServletFileUpload
对象,处理乱码问题setHeaderEncoding
,设置单个文件的最大值setFileSizeMax
,设置总共能够上传文件的大小setSizeMax
,还可选择监听文件上传进度setProgressListener
; - 通过
ServletFileUpload
对象的parseRequest
方法,把前端请求解析,封装成一个FileItem
对象; - 判断上传的文件,是普通的表单,还是带文件的表单
isFormField
,getFieldName
获取前端表单控件的name
; - 通过
FileItem
的getName
方法,获取文件名,并判断是否为空,可以使用UUID
保证文件名唯一性; - 拼接文件真实存在路径,并给每个文件,创建一个对应的文件夹;
- 通过
FileItem
的getInputStream
方法获取文件上传的流,并创建一个文件输出流,将文件写入真实保存路径,最后关闭流; - 上传成功,将页面转发到新的提示页面。
1.4 代码实现
- 前端页面:文件上传
upload.jsp
<form action="${pageContext.request.contextPath}/upload.do" enctype="multipart/form-data" method="post">
<p>上传用户:<input type="text" name="username"></p>
<p>上传文件:<input type="file" name="file1"></p>
<p>上传文件:<input type="file" name="file2"></p>
<p>
<input type="submit" name="submit"> | <input type="reset" name="reset">
</p>
</form>
- servlet:
package com.study.servlet;
import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.ProgressListener;
import org.apache.commons.fileupload.RequestContext;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.UUID;
@WebServlet("/upload.do")
public class UploadServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 判断上传文件是普通表单,还是带文件的表单
if (!ServletFileUpload.isMultipartContent(new Rc(req))) {
// 普通表单,终止方法运行,直接返回
return;
}
// 创建文件保存路径,建议WEB-INF目录下,安全,用户无法直接访问
String uploadPath = this.getServletContext().getRealPath("/WEB-INF/upload");
System.out.println(uploadPath);
File uploadFile = new File(uploadPath);
// 文件夹不存在就创建
if (!uploadFile.exists()) {
uploadFile.mkdir();
}
// 缓存:临时路径,文件超过预期大小,放到临时文件夹中,过几天自动删除,或者提醒用户转存为永久
String tmpPath = this.getServletContext().getRealPath("WEB-INF/tmp");
File file = new File(tmpPath);
if (!file.exists()) {
file.mkdir();
}
// 处理上传的文件,一般通过流来获取,可以使用request.getInputStream(),原生态的文件上传流获取,十分麻烦;
// 建议使用 Apache的文件上传组件来实现,common-fileupload,它需要依赖于 commons-io组件;
// 1、创建DiskFileItemFactory对象,处理文件上传路径或大小的限制
DiskFileItemFactory factory = getDiskFileItemFactory(uploadFile);
// 2、获取ServletFileUpload
ServletFileUpload upload = getServletFileUpload(factory);
// 3、处理上传的文件
try {
String msg = uploadParseRequest(upload, req, uploadPath);
req.setAttribute("msg", msg);
req.getRequestDispatcher("info.jsp").forward(req, resp);
} catch (FileUploadException e) {
e.printStackTrace();
}
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
/**
* 处理文件上传路径或大小的限制:文件大小、编码有默认设置
*
* @param file
* @return
*/
public static DiskFileItemFactory getDiskFileItemFactory(File file) {
DiskFileItemFactory factory = new DiskFileItemFactory();
// 工厂设置:缓冲区大小(上传的文件大于这个缓冲区的时,存放到临时文件中)
// 缓冲区大小为1M,可以不设置,有默认设置
factory.setSizeThreshold(1024 * 1024);
// 临时文件目录,需要一个File
factory.setRepository(file);
return factory;
}
/**
* 获取ServletFileUpload
*
* @param factory
* @return
*/
public static ServletFileUpload getServletFileUpload(DiskFileItemFactory factory) {
ServletFileUpload upload = new ServletFileUpload(factory);
// 可选部分,可不设置==========================
// 监听文件上传进度
upload.setProgressListener(new ProgressListener() {
@Override
// pBytesRead:已经读取到的文件大小
// pContentLength:文件大小
public void update(long pBytesRead, long pContentLength, int pItems) {
System.out.println("总大小:" + pContentLength + "已上传" + pBytesRead);
}
});
// 处理乱码问题
upload.setHeaderEncoding("utf-8");
// 设置单个文件的最大值 10M
upload.setFileSizeMax(1024 * 1024 * 10);
// 设置总共能够上传文件的大小
upload.setSizeMax(1024 * 1024 * 10);
// =====================================
return upload;
}
/**
* 处理上传的文件
*
* @param upload
* @param req
* @param uploadPath
* @return
* @throws FileUploadException
* @throws IOException
*/
public static String uploadParseRequest(ServletFileUpload upload, HttpServletRequest req,
String uploadPath) throws FileUploadException, IOException {
String msg = "";
// 把前端请求解析,封装成一个FileItem对象(tomcat10 req需要处理,非则报错)
List<FileItem> fileItems = upload.parseRequest(new Rc(req));
for (FileItem fileItem : fileItems) {
if (fileItem.isFormField()) {
String name = fileItem.getFieldName();
String value = fileItem.getString("UTF-8");
System.out.println(name + ":" + value);
} else {
//****************************处理文件****************************
// 拿到文件名字
String uploadFileName = fileItem.getName();
System.out.println("上传的文件名:" + uploadFileName);
if (uploadFileName.trim().equals("") || uploadFileName == null) {
continue;
}
// 获得上传的文件名
String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);
// 获得文件的后缀名
String fileExName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);
// 如果文件后缀名fileExName不是所需的直接return,不进行处理,告诉用户文件类型不对
System.out.println("文件信息 [文件名:" + fileName + "---文件类型" + fileExName + "]");
// UUID.randomUUID,随机生一个唯一识别的通用码
// 网络传输中的东西,都需要序列化
// pojo,实体类,如果想要在多个电脑运行,传输--->需要吧对象都序列化了
// JNI=java Native Interface
// implements Serializable :标记接口,JVM--->java栈 本地方法栈 native-->c++
// 可以使用UUID(唯一识别通用码)保证文件名唯一(生成目录)
String uuidPath = UUID.randomUUID().toString();
//****************************处理文件完毕****************************
// 设置存储路径:创建目录/UUID生成目录
String realPath = uploadPath + "/" + uuidPath;
// 给每个文件创建一个对应的文件夹
File realPathFile = new File(realPath);
if (!realPathFile.exists()) {
realPathFile.mkdir();
}
//****************************存放地址完毕*****************************
// 获得文件上传的流
InputStream inputStream = fileItem.getInputStream();
// 创建一个文件输出流
// realPath是真实的文件夹
FileOutputStream fos = new FileOutputStream(realPath + "/" + fileName);
// 创建一个缓冲区
byte[] buffer = new byte[1024 * 1024];
// 判断是否读取完毕
int len = 0;
while ((len = inputStream.read(buffer)) > 0) {
fos.write(buffer, 0, len);
}
// 关闭流
fos.close();
inputStream.close();
msg = "文件上传成功";
// 上传成功,清除临时文件
fileItem.delete();
//*************************文件传输完毕**************************
}
}
return msg;
}
// tomcat10 处理req,非则报错
static class Rc implements RequestContext {
HttpServletRequest request = null;
public Rc(HttpServletRequest request) {
this.request = request;
}
@Override
public String getCharacterEncoding() {
return request.getCharacterEncoding();
}
@Override
public String getContentType() {
return request.getContentType();
}
@Override
public int getContentLength() {
return request.getContentLength();
}
@Override
public InputStream getInputStream() throws IOException {
return request.getInputStream();
}
}
}
- 返回页面:
info.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>消息提示</title>
</head>
<body>
${msg}
</body>
</html>
- 注册 Servlet:配置
web.xml
或通过注解@WebServlet("/upload.do")
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee
https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
version="5.0">
<servlet>
<servlet-name>UploadServlet</servlet-name>
<servlet-class>com.study.servlet.UploadServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>UploadServlet</servlet-name>
<url-pattern>/upload.do</url-pattern>
</servlet-mapping>
</web-app>
-
测试结果:
二、文件下载
2.1 概述
- 文件下载,可以通过 Servlet 类、输入和输出流实现;
- HTTP 协议中设置响应头,告知处理数据的方式为下载方式:
// 二进制文件
resp.addHeader("Content-Type","application/octet-stream");
// 这是一个附件下载
resp.addHeader("Content-Disposition","attachment;filename="+filename);
- 当点击下载链接时,把请求传给 Servelt,在 Servlet 中:
- 获取下载文件的地址,并根据地址,创建文件字节输入流;
- 通过流,读取下载的文件内容,最后将读取的内容,通过输出流写入目标文件中。
2.2 代码实现
- 前端页面:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>文件下载</title>
</head>
<body>
<a href="${pageContext.request.contextPath}/download?filename=1.xlsx">文件下载</a>
</body>
</html>
- Servlet:
@WebServlet("/download")
public class DownloadServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1.从前端获取下载文件名
String fileName = req.getParameter("filename");
// 2.获取下载路径
String realPath = this.getServletContext().getRealPath("/download/" + fileName);
// 3. 设置浏览器的下载支持(Content-Disposition),并将文件名转码,避免乱码
resp.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));
// 4.获取下载文件的输入流
FileInputStream in = new FileInputStream(realPath);
// 5.创建缓冲区
int len = 0;
byte[] buffer = new byte[1024];
// 6.获取 OutputStream 对象
ServletOutputStream out = resp.getOutputStream();
// 7.将 FileOutputStream 流写入到 buffer 缓冲区,使用 OutputStream,将缓冲区中的数据,输出到客户端
while ((len = in.read(buffer)) > 0) {
out.write(buffer, 0, len);
}
// 8.关闭流
in.close();
out.close();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 注册 Servlet:配置
web.xml
或通过注解方式@WebServlet("/download")
三、邮件发送
3.1 邮件的发送与接收原理
-
要实现电子邮件功能,必须有专门的 电子邮件服务器;
-
电子邮箱(E-Mail 地址)的获得,需要在邮件服务器上进行申请:
-
以
QQ邮箱
为例,需要从账户设置中开启POP3/SMTP
服务。
-
-
用户连接邮件服务器,需要遵循一定的通讯规则:
- 发送:SMTP 协议;
- 接收:POP3 协议;
3.2 邮件的发送与接收过程
模拟邮件发送与接收:
-
发送:
xx.@qq.com
通过smtp
协议,连接到 QQ 邮箱的smtp
服务器,并将邮件发送给服务器,接着,QQ 邮箱的smtp
服务器,将邮件 转投 给 163 的smtp
服务器; -
接收:163 的
smtp
服务器,将接收到的邮件,存储到xx.@163.com
的存储空间中,xx.@163.com
通过POP3
协议,连接到 163 的POP3
服务器上,收取邮件,POP3
服务器从xx.@163.com
账号的存储空间中,取出该邮件,并回送给xx.@163.com
账户;
3.3 代码实现邮件的发送
-
用到的包:
-
或 Maven 导入依赖:
<dependency> <groupId>com.sun.mail</groupId> <artifactId>jakarta.mail</artifactId> <version>2.0.1</version> </dependency> <dependency> <groupId>jakarta.activation</groupId> <artifactId>jakarta.activation-api</artifactId> <version>2.1.0</version> </dependency>
-
QQ 邮箱,需获取授权码
- QQ 邮箱 -> 设置 -> 帐户 -> 开启
pop3/smtp
服务 -> 生成授权码(需手机验证);
- QQ 邮箱 -> 设置 -> 帐户 -> 开启
- 简单邮件发送(没有图片和附件)
-
实现过程:
-
创建 session 对象;
-
创建 Transport 对象;
-
使用邮箱的用户名和授权码,连上邮件服务器;
-
创建 Message 对象(需传递 session)message 需要指明发件人、收件人以及文件内容;
-
发送邮件;
-
关闭连接;
-
-
实例:
package com.sutdy.mail;
import com.sun.mail.util.MailSSLSocketFactory;
import jakarta.mail.*;
import jakarta.mail.internet.InternetAddress;
import jakarta.mail.internet.MimeMessage;
import java.util.Properties;
public class MailDemo01 {
public static void main(String[] args) throws Exception {
Properties prop = new Properties();
// 设置邮件服务器
prop.setProperty("mail.host", "smtp服务器地址");
// 邮件发送协议
prop.setProperty("mail.transport.protocol", "smtp");
// 需要验证用户密码
prop.setProperty("mail.smtp.auth", "true");
/* QQ邮箱需要设置SSL加密(其它邮箱不需要)
MailSSLSocketFactory sf = new MailSSLSocketFactory();
sf.setTrustAllHosts(true);
prop.put("mail.smtp.ssl.enable", "true");
prop.put("mail.smtp.ssl.socketFactory", sf);
*/
// 使用javaMail发送邮件的6个步骤
// 1.创建定义整个应用程序所需要的环境信息的session对象
/* QQ邮箱需要,其它不需要
Session session = Session.getDefaultInstance(prop, new Authenticator() {
@Override
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication("邮箱", "授权码");
}
});
*/
Session session = Session.getDefaultInstance(prop);
// 开启session的debug模式,这样可以查看到程序发送Email的运行状态
session.setDebug(true);
// 2.通过session得到transport对象
Transport ts = session.getTransport();
// 3.使用邮箱的用户名和密码连上邮件服务器(QQ邮箱:密码为 授权码)
ts.connect("smtp服务器地址", "邮箱用户名", "邮箱密码");
// 4.创建邮件:写文件
// 注意需要传递session
MimeMessage message = new MimeMessage(session);
// 指明邮件的发件人
message.setFrom(new InternetAddress("发件人邮箱"));
// 指明邮件的收件人
message.setRecipient(Message.RecipientType.TO, new InternetAddress("收件人邮箱"));
// 邮件标题
message.setSubject("标题");
// 邮件的内容:文本(如有附件,图片等,只需更改此处内容)
message.setContent("内容", "text/html;charset=UTF-8");
// 5.发送邮件
ts.sendMessage(message, message.getAllRecipients());
// 6.关闭连接
ts.close();
}
}
- 复杂的邮件发送(带图片和附件)
-
MIME:多用途互联网邮件扩展类型;
-
MimeBodyPart 类:
-
jakarta.mail.internet.MimeBodyPart
类,表示 MIME 消息,和MimeMessage
类一样,从Part
接口继承过来;
-
-
MimeMultipart 类:
-
jakarta.mail.internet.MimeMultipart
类,是抽象类Mulitipart
的实现子类,用来组合多个 MIME 消息,一个MimeMultipart
对象可以包含多个代表 MIME 消息的MimeBodyPart
对象。
-
-
每一个文本、图片、附件,可以分为一个
MimeBodyPart
,由MimeMultipart
完成组装; -
实例:邮件的内容部分,其余代码不变
// ===================邮件的内容:带图片和附件===================
// (1.1)准备图片内容
MimeBodyPart image = new MimeBodyPart();
// 图片需要经过数据化的处理
DataHandler dh = new DataHandler(new FileDataSource("图片路径"));
// 在part中放入,处理过图片的数据
image.setDataHandler(dh);
// 给这个part设置ID名字,后面通过 src的cid使用
image.setContentID("bz.jpg");
// (1.2)准备正文内容
MimeBodyPart text = new MimeBodyPart();
text.setContent("这是一张正文<img src='cid:bz.jpg'>", "text/html;charset=UTF-8");
// (1.3)准备附件内容
MimeBodyPart file = new MimeBodyPart();
file.setDataHandler(new DataHandler(new FileDataSource("文件路径")));
// 给附件设置别名
file.setFileName("1.txt");
// (2.1)描述数据关系
MimeMultipart multipart1 = new MimeMultipart();
multipart1.addBodyPart(image);
multipart1.addBodyPart(text);
// alternative 只能发送 文本
// related 发送文本 + 图片
// mixed 发送文本 + 图片 + 附件
multipart1.setSubType("related");
// (2.2)正文内容:图片+文本
MimeBodyPart contentText = new MimeBodyPart();
contentText.setContent(multipart1);
// (2.3)拼接所有内容:正文 + 附件
MimeMultipart allFile = new MimeMultipart();
// 添加附件
allFile.addBodyPart(file);
// 添加正文内容
allFile.addBodyPart(contentText);
allFile.setSubType("mixed");
// (3)设置到消息中,保存修改
// 把编辑好的邮件放到消息中
message.setContent(allFile);
// 保存修改
message.saveChanges();
// 5.发送邮件
ts.sendMessage(message, message.getAllRecipients());
// 6.关闭连接
ts.close();
- JavaWeb 发送邮件
-
实现:用户注册成功后,系统自动给用户邮箱发送邮件;
-
创建项目,并导入 JavaWeb 所需的相关依赖;
-
项目结构:
-
前端注册页面:
index.jsp
<form action="${pageContext.request.contextPath}/RegisterServlet.do" method="post">
<p>
<label for="username">用户名:</label>
<input type="text" name="username" id="username">
</p>
<p>
<label for="pwd">密码:</label>
<input type="text" name="password" id="pwd">
</p>
<p>
<label for="email">邮箱:</label>
<input type="text" name="email" id="email">
</p>
<p><input type="submit" value="注册"></p>
</form>
- 前端响应页面:
info.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>注册成功</title>
</head>
<body>
${message}
</body>
</html>
- 工具类:邮件发送核心类;
public class Sendmail extends Thread {
// 用于发送邮件的邮箱
private String from = "发件邮箱";
// 邮箱用户名
private String username = "发件邮箱用户名";
// 邮箱密码
private String password = "发件邮箱密码";
// 邮箱服务器 smtp
private String host = "发件邮箱服务器";
// 调用User类
private User user;
public Sendmail(User user) {
this.user = user;
}
@Override
// 重写run方法,在run方法中发送邮件给指定用户
public void run() {
try {
Properties prop = new Properties();
// 设置邮件服务器
prop.setProperty("mail.host", host);
// 邮件发送协议
prop.setProperty("mail.transport.protocol", "smtp");
// 需要验证用户密码
prop.setProperty("mail.smtp.auth", "true");
/* QQ邮箱需要设置SSL加密(其它邮箱不需要)
MailSSLSocketFactory sf = new MailSSLSocketFactory();
sf.setTrustAllHosts(true);
prop.put("mail.smtp.ssl.enable", "true");
prop.put("mail.smtp.ssl.socketFactory", sf);
*/
// 使用javaMail发送邮件的6个步骤
// 1.创建定义整个应用程序所需要的环境信息的session对象
/* QQ邮箱需要,其它不需要
Session session = Session.getDefaultInstance(prop, new Authenticator() {
@Override
protected PasswordAuthentication getPasswordAuthentication() {
return new PasswordAuthentication("发件邮箱", "授权码");
}
});
*/
Session session = Session.getDefaultInstance(prop);
// 开启session的debug模式,这样可以查看到程序发送Email的运行状态
session.setDebug(true);
// 2.通过session得到transport对象
Transport ts = null;
ts = session.getTransport();
// 3.使用邮箱的用户名和授权码连上邮件服务器
ts.connect(host, username, password);
// 4.创建邮件:写文件
// 注意需要传递session
MimeMessage message = new MimeMessage(session);
// 指明邮件的发件人
message.setFrom(new InternetAddress(from));
// 指明邮件的收件人:前端获取 user.getEmail()
message.setRecipient(Message.RecipientType.TO, new InternetAddress(user.getEmail()));
// 邮件标题
message.setSubject("注册通知");
// 邮件的文本内容(只需更换此处内容)
String info = "恭喜您注册成功,您的用户名:" + user.getUsername() + " ,密码:" + user.getPassword();
message.setContent(info, "text/html;charset=UTF-8");
// 5.发送邮件
ts.sendMessage(message, message.getAllRecipients());
// 6.关闭连接
ts.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
- servlet:
@WebServlet("/RegisterServlet.do")
public class RegisterServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 前端获取属性值
String username = req.getParameter("username");
String password = req.getParameter("password");
String email = req.getParameter("email");
// 创建User对象,存储属性值
User user = new User(username, password, email);
// 用户注册成功后,给用户发送邮件
// 使用线程发送邮件,防止出现耗时和网站人多的情况
Sendmail send = new Sendmail(user);
// 启动线程,线程启动后就会执行run方法,发送邮件
send.start();
System.out.println("success");
// 注册用户,响应页面
req.setAttribute("message", "注册成功!已发送邮件,请注意查收");
req.getRequestDispatcher("info.jsp").forward(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
-
注册 Servlet:配置
web.xml
或通过注解方式@WebServlet("/RegisterServlet.do")
-
出现异常:
-
解决方法:将
jar
包,复制到Tomcat
的lib
目录下:
- springboot 实现邮件发送
- 创建 springboot 空项目;
- 导入依赖:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-mail</artifactId>
</dependency>
- 测试类:
@SpringBootTest
class Demo1ApplicationTests {
// 注入Java邮件发送包
@Autowired
JavaMailSenderImpl javaMailSender;
// 普通文本邮件
@Test
void contextLoads() {
// 发送邮件
// 邮件接收人
// 内容
// 创建邮件消息
SimpleMailMessage message = new SimpleMailMessage();
// 设置主题
message.setSubject("SpringBoot 发送邮件");
// 设置文本内容
message.setText("SpringBoot 邮件测试");
// 发送者
message.setFrom("发件人邮箱");
// 接受者
message.setTo("收件人邮箱");
// 发送
javaMailSender.send(message);
}
// 带附件的邮件
@Test
void test() throws Exception {
// 带附件带图片 需要创建发送消息
MimeMessage mimeMessage = javaMailSender.createMimeMessage();
// 创建处理类 处理 mimeMessage true 为更改编码
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage, true);
helper.setSubject("SpringBoot 带附件的邮件");
helper.setText("<h1>SpringBoot 带附件的邮件</h1>", true);
// new File() 指向文件的路径 + 文件名字
helper.addAttachment("测试文档.txt", new File("路径"));
helper.addAttachment("测试图片.png", new File("路径"));
helper.setFrom("发件人邮箱");
helper.setTo("收件人邮箱");
javaMailSender.send(mimeMessage);
}
}
- 编写 springboot 配置文件:
# 邮箱的用户名
spring.mail.username=邮箱的用户名
# 邮箱的密码 QQ邮箱为:授权码
spring.mail.password=邮箱的密码
# smtp服务器
spring.mail.host=smtp服务器
# 需要验证用户密码 设置为安全的
spring.mail.properties.mail.smtyp.ssl.enable=true
网友评论