技术点12:文件的上传和下载
Posted 897463196-a
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了技术点12:文件的上传和下载相关的知识,希望对你有一定的参考价值。
文件的上传和下载
一、文件的上传介绍(*****重点)
步骤:
1、要有一个 form 标签,method=post 请求
2、form 标签的 encType 属性值必须为 multipart/form-data 值
3、在 form 标签中使用 input type=file 添加上传的文件
4、编写服务器代码(Servlet 程序)接收,处理上传的数据。
注:encType=multipart/form-data 表示提交的数据,以多段(每一个表单项一个数据段)的形式进行拼接,然后以二进制流的形式发送给服务器
编写form.jsp页面:
使用UploadServlet程序接收上传的数据:
浏览器访问:
控制台接收到的数据:
1.1、文件上传,HTTP 协议的说明
1.2、commons-fileupload.jar 常用 API 介绍说明
针对于上传过来的数据,我们要解析获取数据内容。Apache为我们提供了这样的jar包,专门用来处理这些上传的数据。
commons-fileupload.jar 需要依赖 commons-io.jar 这个包,所以两个包我们都要引入。
commons-fileupload.jar 和 commons-io.jar 包中,我们常用的类有哪些?
示例:
public class UploadServlet extends HttpServlet {
@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());
System.out.println("表单项的value属性值:" + fileItem.getString("UTF-8"));
}else{
//上传文件表单项
System.out.println("表单项的name属性值:" + fileItem.getFieldName());
System.out.println("上传的文件名:" + fileItem.getName());
//将文件写入磁盘的某个位置
fileItem.write(new File("c:\\" + fileItem.getName()));
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
效果:
二、文件下载
2.1、准备下载的资源
2.2、创建Download.java这个Servlet程序用于下载
package com.zixue.web; import org.apache.commons.io.IOUtils; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; /** * @author Mr Guo * @create 2020-11-13 12:54 */ public class Download extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //1.获取要下载的文件名 String downloadFileName = "1.jpg"; //2.读取要下载的文件内容(通过ServletContext对象可以获取) ServletContext servletContext = getServletContext(); //获取要下载的文件类型 String mimeType = servletContext.getMimeType("/file/" + downloadFileName); System.out.println("下载的文件类型:" + mimeType); //4.在回传前,通过响应头告诉客户端返回的数据类型 resp.setContentType(mimeType); //5.还要告诉客户端收到的数据是用于下载使用(还是使用响应头) // Content-Disposition 响应头,表示收到的数据怎么处理 // attachment 表示附件,表示下载使用 // filename= 表示指定下载的文件名 resp.setHeader("Content-Disposition", "attachment; filename=" + downloadFileName); /** * / 被服务器解析表示的地址为:http://ip:port/工程名/ 映射到代码的web目录 */ InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName); //获取响应的输出流 OutputStream outputStream = resp.getOutputStream(); //3.把下载的文件内容回传给客户端 //读取输入流中全部的数据,复制给输出流,输出给客户端 IOUtils.copy(resourceAsStream,outputStream); } }
2.3、中文名乱码问题解决方案
方案一:URLEncoder 解决 IE 和谷歌浏览器的附件中文名问题
如果客户端浏览器是 IE 浏览器 或者 是谷歌浏览器。我们需要使用 URLEncoder 类先对中文名进行 UTF-8 的编码操作。
因为 IE 浏览器和谷歌浏览器收到含有编码后的字符串后会以 UTF-8 字符集进行解码显示。
resp.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode("中国.jpg", "UTF-8"));
方案二:BASE64 编解码 解决 火狐浏览器的附件中文名问题
BASE64 编解码操作如下:
效果如下:
如果客户端浏览器是火狐浏览器。 那么我们需要对中文名进行 BASE64 的编码操作。
因为火狐使用的是 BASE64 的编解码方式还原响应中的汉字。所以需要使用 BASE64Encoder 类进行编码操作。
综上,我们如何解决不同浏览器的兼容性问题呢?对于火狐浏览器,使用BASE64进行编码,其它,则使用URLEncoder进行编码。怎么区分不同的浏览器呢?在请求头中,有一个User-Agent请求头,判断是否包含Firefox进行区分。最终效果如下:
package com.zixue.web; import org.apache.commons.io.IOUtils; import sun.misc.BASE64Encoder; import javax.servlet.ServletContext; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.URLEncoder; /** * @author Mr Guo * @create 2020-11-13 12:54 */ public class Download extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { //1.获取要下载的文件名 String downloadFileName = "1.jpg"; //2.读取要下载的文件内容(通过ServletContext对象可以获取) ServletContext servletContext = getServletContext(); //获取要下载的文件类型 String mimeType = servletContext.getMimeType("/file/" + downloadFileName); System.out.println("下载的文件类型:" + mimeType); //4.在回传前,通过响应头告诉客户端返回的数据类型 resp.setContentType(mimeType); //5.还要告诉客户端收到的数据是用于下载使用(还是使用响应头) // Content-Disposition 响应头,表示收到的数据怎么处理 // attachment 表示附件,表示下载使用 // filename= 表示指定下载的文件名 if (req.getHeader("User-Agent").contains("Firefox")){ resp.setHeader("Content-Disposition", "attachment; filename==?UTF-8?B?"+ new BASE64Encoder().encode("中国.jpg".getBytes("UTF-8")) +"?="); }else{ resp.setHeader("Content-Disposition", "attachment; filename=" + URLEncoder.encode("中国.jpg", "UTF-8")); } /** * / 被服务器解析表示的地址为:http://ip:port/工程名/ 映射到代码的web目录 */ InputStream resourceAsStream = servletContext.getResourceAsStream("/file/" + downloadFileName); //获取响应的输出流 OutputStream outputStream = resp.getOutputStream(); //3.把下载的文件内容回传给客户端 //读取输入流中全部的数据,复制给输出流,输出给客户端 IOUtils.copy(resourceAsStream,outputStream); } }
以上是关于技术点12:文件的上传和下载的主要内容,如果未能解决你的问题,请参考以下文章
从本地上传到服务器上,再下载下来打开就成了乱码。本地上传的文档没有问题。