Java web----文件上传

来源:互联网 发布:.net中级程序员面试题 编辑:程序博客网 时间:2024/05/22 03:09

1 文件上传对页面的要求

  • 必须使用表单,而不能是超链接;
  • 表单的method必须是POST,而不能是GET;
  • 表单的enctype必须是multipart/form-data;
  • 在表单中添加file表单字段,即<input type=”file”…/>

  <body>     <form action="<c:url value='/UploadServlet01'/>" method="post" enctype="multipart/form-data">     姓名:<input type="text" name="username"><br/>     照片:<input type="file" name="phone"><br/>          <input type="submit" value="提交"/>     </form>  </body>

2 比对文件上传表单和普通文本表单的区别

通过httpWatch查看“文件上传表单”和“普通文本表单”的区别。

文件上传表单的enctype=”multipart/form-data”,表示多部件表单数据;

普通文本表单可以不设置enctype属性:

  • 当method=”post”时,enctype的默认值为application/x-www-form-urlencoded,表示使用url编码正文;
  • 当method=”get”时,enctype的默认值为null,没有正文,所以就不需要enctype了。

具体不同可以抓包查看

3 文件上传对Servlet的要求

首先我们要肯定一点,文件上传表单的数据也是被封装到request对象中的。

request.getParameter(String)方法获取指定的表单字段字符内容,但文件上传表单已经不在是字符内容,而是字节内容,所以失效。 

这时可以使用request的getInputStream()方法获取ServletInputStream对象,它是InputStream的子类,这个ServletInputStream对象对应整个表单的正文部分(从第一个分隔线开始,到最后),这说明我们需要的解析流中的数据。当然解析它是很麻烦的一件事情,而Apache已经帮我们提供了解析它的工具:commons-fileupload。

package com.cug.upload;import java.io.IOException;import javax.servlet.ServletException;import javax.servlet.ServletInputStream;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import org.apache.commons.io.IOUtils;public class UploadServlet01 extends HttpServlet{@Overrideprotected void doPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {req.setCharacterEncoding("utf-8");resp.setContentType("text/html;charset=utf-8");ServletInputStream in = req.getInputStream();String s = IOUtils.toString(in);System.out.println(s);}}

4 commons-fileupload

为什么使用fileupload:

上传文件的要求比较多,需要记一下:

  • 必须是POST表单;
  • 表单的enctype必须是multipart/form-data;
  • 在表单中添加file表单字段,即<input type=”file”…/> 

Servlet的要求:

  • 不能再使用request.getParameter()来获取表单数据;
  • 可以使用request.getInputStream()得到所有的表单数据,而不是一个表单项的数据;
  • 这说明不使用fileupload,我们需要自己来对request.getInputStream()的内容进行解析!!!

4.1 fileupload概述

fileupload是由apache的commons组件提供的上传组件。它最主要的工作就是帮我们解析request.getInputStream()。

fileupload组件需要的JAR包有:

  • commons-fileupload.jar,核心包;
  • commons-io.jar,依赖包。

4.2 fileupload简单应用

fileupload的核心类有:DiskFileItemFactory、ServletFileUpload、FileItem。

使用fileupload组件的步骤如下:

  • 创建工厂类DiskFileItemFactory对象:DiskFileItemFactoryfactory = new DiskFileItemFactory()
  • 使用工厂创建解析器对象:ServletFileUpload fileUpload = new ServletFileUpload(factory)
  • 使用解析器来解析request对象:List<FileItem>list = fileUpload.parseRequest(request)

隆重介绍FileItem类,它才是我们最终要的结果。一个FileItem对象对应一个表单项(表单字段)。一个表单中存在文件字段和普通字段,可以使用FileItem类的isFormField()方法来判断表单字段是否为普通字段,如果不是普通字段,那么就是文件字段了。

  • String getName():获取文件字段的文件名称;
  • String getString():获取字段的内容,如果是文件字段,那么获取的是文件内容,当然上传的文件必须是文本文件;
  • String getFieldName():获取字段名称,例如:<inputtype=”text” name=”username”/>,返回的是username;
  • String getContentType():获取上传的文件的类型,例如:text/plain。
  • int getSize():获取上传文件的大小;
  • boolean isFormField():判断当前表单字段是否为普通文本字段,如果返回false,说明是文件字段;
  • InputStream getInputStream():获取上传文件对应的输入流;
  • void write(File):把上传的文件保存到指定文件中。

4.3 简单上传示例

 <form action="${pageContext.request.contextPath }/FileUploadServlet" method="post" enctype="multipart/form-data">    用户名:<input type="text" name="username"/><br/>    文件1:<input type="file" name="file1"/><br/>    <input type="submit" value="提交"/>    </form>

public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {// 因为要使用response打印,所以设置其编码response.setContentType("text/html;charset=utf-8");// 创建工厂DiskFileItemFactory dfif = new DiskFileItemFactory();// 使用工厂创建解析器对象ServletFileUpload fileUpload = new ServletFileUpload(dfif);try {// 使用解析器对象解析request,得到FileItem列表List<FileItem> list = fileUpload.parseRequest(request);// 遍历所有表单项for(FileItem fileItem : list) {// 如果当前表单项为普通表单项if(fileItem.isFormField()) {// 获取当前表单项的字段名称String fieldName = fileItem.getFieldName();// 如果当前表单项的字段名为usernameif(fieldName.equals("username")) {// 打印当前表单项的内容,即用户在username表单项中输入的内容response.getWriter().print("用户名:" + fileItem.getString() + "<br/>");}} else {//如果当前表单项不是普通表单项,说明就是文件字段String name = fileItem.getName();//获取上传文件的名称// 如果上传的文件名称为空,即没有指定上传文件if(name == null || name.isEmpty()) {continue;}// 获取真实路径,对应${项目目录}/uploads,当然,这个目录必须存在String savepath = this.getServletContext().getRealPath("/uploads");// 通过uploads目录和文件名称来创建File对象File file = new File(savepath, name);// 把上传文件保存到指定位置fileItem.write(file);// 打印上传文件的名称response.getWriter().print("上传文件名:" + name + "<br/>");// 打印上传文件的大小response.getWriter().print("上传文件大小:" + fileItem.getSize() + "<br/>");// 打印上传文件的类型response.getWriter().print("上传文件类型:" + fileItem.getContentType() + "<br/>");}}} catch (Exception e) {throw new ServletException(e);} }
5 文件上传之细节

5.1  把上传的文件放到WEB-INF目录下

如果没有把用户上传的文件存放到WEB-INF目录下,那么用户就可以通过浏览器直接访问上传的文件,这是非常危险的。

假如说用户上传了一个a.jsp文件,然后用户在通过浏览器去访问这个a.jsp文件,那么就会执行a.jsp中的内容,如果在a.jsp中有如下语句:Runtime.getRuntime().exec(“shutdown –s –t 1”);,那么你就会…

 

通常我们会在WEB-INF目录下创建一个uploads目录来存放上传的文件,而在Servlet中找到这个目录需要使用ServletContext的getRealPath(String)方法,例如在我的upload1项目中有如下语句:

ServletContextservletContext = this.getServletContext();

String savepath= servletContext.getRealPath(“/WEB-INF/uploads”); 

其中savepath为:F:\tomcat6_1\webapps\upload1\WEB-INF\uploads。

5.2 文件名称(完整路径、文件名称)

使用不同浏览器测试,其中IE6就会返回上传文件的完整路径,不知道IE6在搞什么,这给我们带来了很大的麻烦,就是需要处理这一问题。

处理这一问题也很简单,无论是否为完整路径,我们都去截取最后一个“\\”后面的内容就可以了。

String name = file1FileItem.getName();int lastIndex = name.lastIndexOf("\\");//获取最后一个“\”的位置if(lastIndex != -1) {//注意,如果不是完整路径,那么就不会有“\”的存在。name = name.substring(lastIndex + 1);//获取文件名称}response.getWriter().print(name);
5.3 中文乱码问题

上传文件名称中包含中文

当上传的谁的名称中包含中文时,需要设置编码,commons-fileupload组件为我们提供了两种设置编码的方式:

  • request.setCharacterEncoding(String):这种方式是我们最为熟悉的方式了;
  • fileUpload.setHeaderEncdoing(String):这种方式的优先级高与前一种。

上传文件的文件内容包含中文:

通常我们不需关心上传文件的内容,因为我们会把上传文件保存到硬盘上!也就是说,文件原来是什么样子,到服务器这边还是什么样子!

但是如果你有这样的需求,非要在控制台显示上传的文件内容,那么你可以使用fileItem.getString(“utf-8”)来处理编码。

文本文件内容和普通表单项内容使用FileItem类的getString(“utf-8”)来处理编码。


5.4  上传文件同名问题(文件重命名)

通常我们会把用户上传的文件保存到uploads目录下,但如果用户上传了同名文件呢?这会出现覆盖的现象。处理这一问题的手段是使用UUID生成唯一名称,然后再使用“_”连接文件上传的原始名称。

例如用户上传的文件是“我的一寸照片.jpg”,在通过处理后,文件名称为:“891b3881395f4175b969256a3f7b6e10_我的一寸照片.jpg”,这种手段不会使文件丢失扩展名,并且因为UUID的唯一性,上传的文件同名,但在服务器端是不会出现同名问题的。

public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {request.setCharacterEncoding("utf-8");DiskFileItemFactory dfif = new DiskFileItemFactory();ServletFileUpload fileUpload = new ServletFileUpload(dfif);try {List<FileItem> list = fileUpload.parseRequest(request);//获取第二个表单项,因为第一个表单项是username,第二个才是file表单项FileItem fileItem = list.get(1);String name = fileItem.getName();//获取文件名称// 如果客户端使用的是IE6,那么需要从完整路径中获取文件名称int lastIndex = name.lastIndexOf("\\");if(lastIndex != -1) {name = name.substring(lastIndex + 1);}// 获取上传文件的保存目录String savepath = this.getServletContext().getRealPath("/WEB-INF/uploads");String uuid = CommonUtils.uuid();//生成uuidString filename = uuid + "_" + name;//新的文件名称为uuid + 下划线 + 原始名称//创建file对象,下面会把上传文件保存到这个file指定的路径//savepath,即上传文件的保存目录//filename,文件名称File file = new File(savepath, filename);// 保存文件fileItem.write(file);} catch (Exception e) {throw new ServletException(e);} }

5.5 一个目录不能存放过多的文件(存放目录打散)

 我们这里使用hash算法来打散:

  • 获取文件名称的hashCode:int hCode = name.hashCode();;
  • 获取hCode的低4位,然后转换成16进制字符;
  •  获取hCode的5~8位,然后转换成16进制字符;
  • 使用这两个16进制的字符生成目录链。例如低4位字符为“5”

  这种算法的好处是,在uploads目录下最多生成16个目录,而每个目录下最多再生成16个目录,即256个目录,所有上传的文件都放到这256个目录下。如果每个目录上限为1000个文件,那么一共可以保存256000个文件。 

例如上传文件名称为:新建文本文档.txt,那么把“新建 文本文档.txt”的哈希码获取到,再获取哈希码的低4位,和5~8位。假如低4位为:9,5~8位为1,那么文件的保存路径为uploads/9/1/。

int hCode = name.hashCode();//获取文件名的hashCode//获取hCode的低4位,并转换成16进制字符串String dir1 = Integer.toHexString(hCode & 0xF);//获取hCode的低5~8位,并转换成16进制字符串String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);//与文件保存目录连接成完整路径savepath = savepath + "/" + dir1 + "/" + dir2;//因为这个路径可能不存在,所以创建成File对象,再创建目录链,确保目录在保存文件之前已经存在new File(savepath).mkdirs();
5.6 上传的单个文件的大小限制

限制上传文件的大小很简单,ServletFileUpload类的setFileSizeMax(long)就可以了。参数就是上传文件的上限字节数,例如servletFileUpload.setFileSizeMax(1024*10)表示上限为10KB。

一旦上传的文件超出了上限,那么就会抛出FileUploadBase.FileSizeLimitExceededException异常。我们可以在Servlet中获取这个异常,然后向页面输出“上传的文件超出限制”。

public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {request.setCharacterEncoding("utf-8");DiskFileItemFactory dfif = new DiskFileItemFactory();ServletFileUpload fileUpload = new ServletFileUpload(dfif);// 设置上传的单个文件的上限为10KBfileUpload.setFileSizeMax(1024 * 10); try {List<FileItem> list = fileUpload.parseRequest(request);//获取第二个表单项,因为第一个表单项是username,第二个才是file表单项FileItem fileItem = list.get(1);String name = fileItem.getName();//获取文件名称// 如果客户端使用的是IE6,那么需要从完整路径中获取文件名称int lastIndex = name.lastIndexOf("\\");if(lastIndex != -1) {name = name.substring(lastIndex + 1);}// 获取上传文件的保存目录String savepath = this.getServletContext().getRealPath("/WEB-INF/uploads");String uuid = CommonUtils.uuid();//生成uuidString filename = uuid + "_" + name;//新的文件名称为uuid + 下划线 + 原始名称int hCode = name.hashCode();//获取文件名的hashCode//获取hCode的低4位,并转换成16进制字符串String dir1 = Integer.toHexString(hCode & 0xF);//获取hCode的低5~8位,并转换成16进制字符串String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);//与文件保存目录连接成完整路径savepath = savepath + "/" + dir1 + "/" + dir2;//因为这个路径可能不存在,所以创建成File对象,再创建目录链,确保目录在保存文件之前已经存在new File(savepath).mkdirs();//创建file对象,下面会把上传文件保存到这个file指定的路径//savepath,即上传文件的保存目录//filename,文件名称File file = new File(savepath, filename);// 保存文件fileItem.write(file);} catch (Exception e) {// 判断抛出的异常的类型是否为FileUploadBase.FileSizeLimitExceededException// 如果是,说明上传文件时超出了限制。if(e instanceof FileUploadBase.FileSizeLimitExceededException) {// 在request中保存错误信息request.setAttribute("msg", "上传失败!上传的文件超出了10KB!");// 转发到index.jsp页面中!在index.jsp页面中需要使用${msg}来显示错误信息request.getRequestDispatcher("/index.jsp").forward(request, response);return;} throw new ServletException(e);} }
5.7 上传文件的总大小限制

有时我们需要限制一个请求的大小。也就是说这个请求的最大字节数(所有表单项之和)!实现这一功能也很简单,只需要调用ServletFileUpload类的setSizeMax(long)方法即可。

例如fileUpload.setSizeMax(1024 * 10);,显示整个请求的上限为10KB。当请求大小超出10KB时,ServletFileUpload类的parseRequest()方法会抛出FileUploadBase.SizeLimitExceededException异常。


5.8 缓存大小与临时目录
public void doPost(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException {request.setCharacterEncoding("utf-8");<strong>DiskFileItemFactory dfif = new DiskFileItemFactory(1024*20, new File("F:\\temp"));</strong>ServletFileUpload fileUpload = new ServletFileUpload(dfif);try {List<FileItem> list = fileUpload.parseRequest(request);FileItem fileItem = list.get(1);String name = fileItem.getName();String savepath = this.getServletContext().getRealPath("/WEB-INF/uploads");// 保存文件fileItem.write(path(savepath, name));} catch (Exception e) {throw new ServletException(e);} }private File path(String savepath, String filename) {// 从完整路径中获取文件名称int lastIndex = filename.lastIndexOf("\\");if(lastIndex != -1) {filename = filename.substring(lastIndex + 1);}// 通过文件名称生成一级、二级目录int hCode = filename.hashCode();String dir1 = Integer.toHexString(hCode & 0xF);String dir2 = Integer.toHexString(hCode >>> 4 & 0xF);savepath = savepath + "/" + dir1 + "/" + dir2;// 创建目录new File(savepath).mkdirs();// 给文件名称添加uuid前缀String uuid = CommonUtils.uuid();filename = uuid + "_" + filename;// 创建文件完成路径return new File(savepath, filename);}


1 0
原创粉丝点击