文件的上传和下载
- 1.文件的上传介绍(重点)
- 1.1 文件上传时候http协议内容
- 1.2 commons-fileupload.jar 常用api介绍说明
- 1.3 fileupload类库的使用
- 2.文件下载
- 代码
- 附件中文名乱码问题解决方案:
- 方案一:URLEncoder 解决 IE 和谷歌浏览器的 附件中 文名问题。
- 方案二:BASE64 编解码 解决 火狐浏览器的附件中文名问题
- BASE64 编解码操作:
文件的上传和下载:
文件的上传和下载,是非常常见的功能,很多的系统中,或者软件中都经常使用文件的上传和下载,比如: qq头像,就是用了文件的上传。
邮箱中也有邮件的上传和下载功能
OA系统中审批有附件材料的上传。
1.文件的上传介绍(重点)
1.要有一个form标签 method=post请求
2.form标签的encType属性值必须为multipart/form-data值
3.再form标签中使用input type=file添加上传的文件
4.滨蟹服务器(servlet程序 )代码接收,处理上传的数据
encType=multipart/form-data 表示提交的数据,以多段(每一个表单项每一个数据段)的形式进行拼接,然后以二进制流的形式送给服务器,
1.1 文件上传时候http协议内容
请求头:
POST /uploadservlet HTTP/1.1
Host: localhost
Connection: keep-alive
Content-Length: 528
Cache-Control: max-age=0
sec-ch-ua: “Google Chrome”;v=“93”, " Not;A Brand";v=“99”, “Chromium”;v=“93”
sec-ch-ua-mobile: ?0
sec-ch-ua-platform: “Windows”
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/93.0.4577.82 Safari/537.36
Origin: http://localhost
Content-Type: multipart/form-data; boundary=----WebKitFormBoundaryUX2GBd81fqG7vZCF
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,/;q=0.8,application/signed-exchange;v=b3;q=0.9
Sec-Fetch-Site: same-origin
Sec-Fetch-Mode: navigate
Sec-Fetch-User: ?1
Sec-Fetch-Dest: document
Referer: http://localhost/upload.jsp
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9
Cookie: Idea-c372728b=41df1464-46ec-4bba-b8c1-d85a2c936a1f; JSESSIONID=A5E2A45B78A84949C32D60B6E7821B5D
请求体
-----------------------------7e51416b09d4
Content-Disposition: form-data; name=“username”
1111
-----------------------------7e51416b09d4
Content-Disposition: form-data; name=“photo”; filename=“C:\Users\26523\Desktop\HTMLä¸æ–‡æ‰‹å†Œ.chm”
Content-Type: application/octet-stream
ITSF
1.2 commons-fileupload.jar 常用api介绍说明
commons-fileupload.jar需要依赖commons-io.jar这个包,所以两个包我们都要引入
第一步:就是需要导入两个jar包
commons-fileupload-1.4.jar
commons-io-2.9.0.jar
commons-fileupload.jar 和 commons-io.jar 包中,我们常用的类有哪些?
ServletFileUpload 类,用于解析上传的数据。
FileItem 类,表示每一个表单项。
boolean ServletFileUpload.isMultipartContent(HttpServletRequest request);
判断当前上传的数据格式是否是多段的格式。
public List parseRequest(HttpServletRequest request)
解析上传的数据
boolean FileItem.isFormField()
判断当前这个表单项,是否是普通的表单项。还是上传的文件类型。
true 表示普通类型的表单项
false 表示上传的文件类型
String FileItem.getFieldName()
获取表单项的 name 属性值
String FileItem.getString()
获取当前表单项的值。
String FileItem.getName();
获取上传的文件名
void FileItem.write( file );
将上传的文件写到 参数 file 所指向抽硬盘位置 。
1.3 fileupload类库的使用
上传文件的表单:
<form action="/uploadservlet" method="post" enctype="multipart/form-data" >
用户名:<input type="text" name = "username"/><br/>
头像:<input type="file" name="photo"/><br/>
<input type="submit" value="上传"/>
uploadServlet.java
package com.atguigu.servlet;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileItemFactory;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
import javax.servlet.ServletException;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.util.List;
public class UploadServlet extends HttpServlet {
/**
*用来处理文件上传
* @param req
* @param resp
* @throws ServletException
* @throws IOException
*/
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1.先判断上传来的数据是否为多段数据(只有多段的数据,才是文件上传的)
if(ServletFileUpload.isMultipartContent(req)){
//创建FileItemFactory工厂实现类
FileItemFactory fileItemFactory = new DiskFileItemFactory();
//创建用于解析上传数据的工具类ServletFileUpload
ServletFileUpload servletFileUpload = new ServletFileUpload(fileItemFactory);
try {
//解析上传的数据,得到每一个表单项fileitem
List<FileItem> list = servletFileUpload.parseRequest(req);
for(FileItem fileItem : list){
if(fileItem.isFormField()){
//普通表单项
System.out.println("表单项的name属性值"+ fileItem.getFieldName());
//参数UTF-8 解决乱码问题
System.out.println("表单项的value属性值:"+ fileItem.getString("UTF-8"));
}else{
//上传的文件
System.out.println("上传文件的文件名"+fileItem.getFieldName());
System.out.println("上传的文件名"+fileItem.getName());
fileItem.write(new File("C:\\Users\\26523\\Desktop\\test\\"+fileItem.getName()));
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
2.文件下载
代码
![在这里插入图片描述]()
web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0">
<context-param>
<param-name>username</param-name>
<param-value>root</param-value>
</context-param>
<context-param>
<param-name>mysql</param-name>
<param-value>jdbc:mysql://localhost:3306</param-value>
</context-param>
<servlet>
<servlet-name>uploadServlet</servlet-name>
<servlet-class>com.atguigu.servlet.UploadServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>uploadServlet</servlet-name>
<url-pattern>/uploadservlet</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>download</servlet-name>
<servlet-class>com.atguigu.servlet.download</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>download</servlet-name>
<url-pattern>/download</url-pattern>
</servlet-mapping>
</web-app>
download.java
package com.atguigu.servlet;
import org.apache.commons.io.IOUtils;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
public class download extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1.获取要下载的文件名
String downloadFileName = "seele1.jpeg";
// 2.读取要下载的文件内容 (通过servletContext对象可以读取)
ServletContext servletContext = getServletContext();
// /斜杠被服务器解析表示地址为http://ip:port/工程名/ 映射到代码的web目录
InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName);
// 4.在回传前,通过响应头告诉客户端返回的数据类型
//获取要下载的文件类型
String mimeType = servletContext.getMimeType("/file/" + downloadFileName);
System.out.println(mimeType);
resp.setContentType(mimeType);
// 5.还要告诉客户端收到的数据是否用于下载使用(还是使用响应头)
//attachment表示附件的意思,表示下载使用
// filename-表示指定下载的文件名
resp.setHeader("content-Disposition","attachment;filename="+downloadFileName);
// 3.把下载的文件内容回传给客户端
//获取响应的输出流
ServletOutputStream outputStream = resp.getOutputStream();
//读取输入流种全部的数据,复制给输出流,输出给客户端
IOUtils.copy(resourceAsStream,outputStream);
}
}
附件中文名乱码问题解决方案:
方案一:URLEncoder 解决 IE 和谷歌浏览器的 附件中 文名问题。
如果客户端浏览器是 IE 浏览器 或者 是谷歌浏览器。
我们需要使用 URLEncoder 类先对中文名进行 UTF-8 的编码 操作。因为 IE 浏览器和谷歌浏览器收到含有编码后的字符串后会以 UTF-8 字符集进行解码显示。
// 把中文名进行 UTF-8 编码操作。
String str = "attachment; fileName=" + URLEncoder.encode("中文.jpg", "UTF-8");
// 然后把编码后的字符串设置到响应头中 response.setHeader("Content-Disposition", str);
方案二:BASE64 编解码 解决 火狐浏览器的附件中文名问题
如果客户端浏览器是火狐浏览器。
那么我们需要对中文名进行 BASE64 的编码操作。
这时候需要把请求头 Content-Disposition: attachment; filename=中文名 编码成为:Content-Disposition: attachment; filename==?charset?B?xxxxx?= =?charset?B?xxxxx?= 现在我们对这段内容进行一下说明。
**=? 表示编码内容的开始
charset 表示字符集
B 表示 BASE64 编码
xxxx 表示文件名 BASE64 编码后的内容
?= 表示编码内容的结束
BASE64 编解码操作:
因为火狐使用的是 BASE64 的编解码方式还原响应中的汉字。
所以需要使用 BASE64Encoder 类进行编码操作。**
// 使用下面的格式进行 BASE64 编码后
String str = "attachment; fileName=" + "=?utf-8?B?" + new BASE64Encoder().encode("中文.jpg".getBytes("utf-8")) + "?=";
// 设置到响应头中
response.setHeader("Content-Disposition", str);
那么我们如何解决上面两种不同编解码方式呢。我们只需要通过判断请求头中 User-Agent 这个请求头携带过来的 浏览器信息即可判断出是什么浏览器。如下:
String ua = request.getHeader("User-Agent");
// 判断是否是火狐浏览器
if (ua.contains("Firefox"))
{ // 使用下面的格式进行 BASE64 编码后
String str = "attachment; fileName=" + "=?utf-8?B?" + new BASE64Encoder().encode("中文.jpg".getBytes("utf-8")) + "?=";
// 设置到响应头中 response.setHeader("Content-Disposition", str);
} else { // 把中文名进行 UTF-8 编码操作。
String str = "attachment; fileName=" + URLEncoder.encode("中文.jpg", "UTF-8");
// 然后把编码后的字符串设置到响应头中 response.setHeader("Content-Disposition", str); }