文件上传分析

1.普通表单提交默认enctype=“application/x-www-form-urlencoded”;但是当表单中存在文件类型时,需要设置enctype=“multipart/form-data”,它不对字符进行编码,用于发送二进制的文件(即所有文件类型,如视频、图片、音乐、文档都可以用此类型entype);还有一种enctype="text/plain"用于发送纯文本内容。

2.表单请求方式必须为post。

3.接收时不能再用request.getParameter(),而是request.getInputStream()解析二进制流,得到ServletInputStream对象。

接下来我们来看看上传一个t文件接收到的二进制流转化为字符串是什么:

(1)表单提交页面:

<%@ page contentType=“text/html;charset=UTF-8” language=“java” %>


练习 (2)接收文件流UploadServlet:

import javax.servlet.ServletException;
 import javax.servlet.annotation.WebServlet;
 import javax.servlet.http.HttpServlet;
 import javax.servlet.http.HttpServletRequest;
 import javax.servlet.http.HttpServletResponse;
 import java.io.IOException;
 import java.io.InputStream;@WebServlet("/UploadServlet")
 public class UploadServlet extends HttpServlet {
 protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
 //获取复杂表单的输入流
 InputStream in=request.getInputStream();//输入流转化为字符串
    byte[] b=new byte[1024];
    in.read(b);
    System.out.println(new String(b));
}

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    doPost(request,response);
}}


运行项目:

(1)表单填写:

test.txt中的内容是:

hello world
好好学习,天天向上
(2)点击提交,运行结果:

------WebKitFormBoundaryzRSPAU9UKnMzdAUZ
Content-Disposition: form-data; name=“username”

hello 灏忛奔
------WebKitFormBoundaryzRSPAU9UKnMzdAUZ
Content-Disposition: form-data; name=“pwd”

12345
------WebKitFormBoundaryzRSPAU9UKnMzdAUZ
Content-Disposition: form-data; name=“pic”; filename=“test.txt”
Content-Type: text/plain

锘縣ello world
濂藉ソ瀛︿範锛屽ぉ澶╁悜涓?

------WebKitFormBoundaryzRSPAU9UKnMzdAUZ–
分析:------WebKitFormBoundaryzRSPAU9UKnMzdAUZ–是分隔符,用于分隔表单的每一个字段。运行结果中有中文乱码后续会处理。将二进制流转化为字符串除了上面用的read()方法,还可以用org.apache.commons.io.IOUtils.toString(in),不过要导入jar包。

FileUpload文件上传

1.FileUpload分析

fileUpload是apache的commons组件提供的上传组件,它最主要的工作就是帮我们解析request.getInpustream()。可以参考在线API文档:

使用fileUpload组件首先需要引入两个jar包:

commons-fileUpload.jar
commons-io.jar
fileUpload的核心类有DiskFileItemFactory、ServletFileUpload、FileItem。

使用fileUpload固定步骤:

创建工厂类:DiskFileItemFactory factory=new DiskFileItemFactory();
创建解析器:ServletFileUpload upload=new ServletFileUpload(factory);
使用解析器解析request对象:List list=upload.parseRequest(request);
一个FileItem对象对应一个表单项。FileItem类有如下方法:

String getFieldName():获取表单项的name的属性值。
String getName():获取文件字段的文件名。如果是普通字段,则返回null
String getString():获取字段的内容。如果是普通字段,则是它的value值;如果是文件字段,则是文件内容。
String getContentType():获取上传的文件类型,例如text/plain、image。如果是普通字段,则返回null。
long getSize():获取字段内容的大小,单位是字节。
boolean isFormField():判断是否是普通表单字段,若是,返回true,否则返回false。
InputStream getInputStream():获得文件内容的输入流。如果是普通字段,则返回value值的输入流。

表单提交页面同上,修改UploadServlet的代码如下:

import org.apache.commons.fileupload.FileItem;
 import org.apache.commons.fileupload.FileUploadException;
 import org.apache.commons.fileupload.disk.DiskFileItemFactory;
 import org.apache.commons.fileupload.servlet.ServletFileUpload;
 import sun.misc.IOUtils;import javax.servlet.ServletException;
 import javax.servlet.annotation.WebServlet;
 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.util.List;@WebServlet("/UploadServlet")
 public class UploadServlet extends HttpServlet {
 protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
 DiskFileItemFactory factory=new DiskFileItemFactory();
 ServletFileUpload upload=new ServletFileUpload(factory);
 try {
 List list=upload.parseRequest(request);
 for (FileItem fileItem:list){
 System.out.println(“fieldName:”+fileItem.getFieldName());
 System.out.println(“name:”+fileItem.getName());
 System.out.println(“string:”+fileItem.getString());
 System.out.println(“contentType:”+fileItem.getContentType());
 System.out.println(“size:”+fileItem.getSize()+“byte”);
 System.out.println(“isFieldForm:”+fileItem.isFormField());
 System.out.println(“inputStream:”+ org.apache.commons.io.IOUtils.toString(fileItem.getInputStream()));
 System.out.println("*************");
 }
 } catch (FileUploadException e) {
 e.printStackTrace();
 }
 }protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    doPost(request,response);
}}


运行结果:

(1)填写表单:

(2)test.txt同上,点击提交结果:

fieldName:username
name:null
string:hello ?°?é±?
contentType:null
size:12byte
isFieldForm:true
inputStream:hello 灏忛奔


fieldName:pwd
name:null
string:12345
contentType:null
size:5byte
isFieldForm:true
inputStream:12345


fieldName:pic
name:test.txt
string:???hello world
??? ?¤??¤???

contentType:text/plain
size:43byte
isFieldForm:false
inputStream:锘縣ello world
濂藉ソ瀛︿範 澶╁ぉ鍚戜笂


2.FileUpload实现文件上传

使用fileUpload组件实现文件上传除了上面的那些方法之外还要注意的:

文件名中文乱码处理:servletFileUpload.setHeaderEncoding(“utf-8”) 或 request.setCharacterEncoding(“utf-8”);
表单普通字段中文乱码处理:new String(str.getBytes(“iso-8859-1”,“utf-8”));
设置内存缓冲区的大小,默认为10KB:diskFileItemFactory.setSizeThreshold(10241024);
指定临时文件目录,如果单个文件的大小超过内存缓冲区,该文件将会临时缓存在此目录下:diskFileItemFactory.setRepository(file);
设置单个文件大小限制,如果有某个文件超过此大小,将抛出FileUploadBase.FileSizeLimitExceededException:servletFileUpload.setFileSizeMax(1024
102410);
设置所有文件,也就是请求大小限制,如果文件总和超过此大小,将抛出FileUploadBase.SizeLimitExceededException:servletFileUpload.setSizeMax(1024
1024*20);
利用UUID生成伪随机字符串作为文件名避免重复:UUID.randomUUID().toString();
将文件写到硬盘上。写完之后,系统会自动将放在临时文件目录的该文件删除:fileItem.write(new File(path,fileName));
注:如果没有指定临时文件目录,默认采用系统默认的临时文件路径,可以通过System.getProperty(“java.io.tmpdir”)获取,Tomcat系统默认临时目录为“<tomcat安装目录>/temp/”。

Apache文件上传组件在解析上传数据中的每个字段内容时,需要临时保存解析出的数据,以便在后面进行数据的进一步处理(保存在磁盘特定位置或插入数据库)。因为Java虚拟机默认可以使用的内存空间是有限的,超出限制时将会抛出“java.lang.OutOfMemoryError”错误。如果上传的文件很大,例如800M的文件,在内存中将无法临时保存该文件内容,Apache文件上传组件转而采用临时文件来保存这些数据;但如果上传的文件很小,例如600个字节的文件,显然将其直接保存在内存中性能会更加好些。

表单提交页面不变,UploadServlet代码如下:

import org.apache.commons.fileupload.FileItem;
 import org.apache.commons.fileupload.FileUploadBase;
 import org.apache.commons.fileupload.FileUploadException;
 import org.apache.commons.fileupload.disk.DiskFileItemFactory;
 import org.apache.commons.fileupload.servlet.ServletFileUpload;
 import sun.misc.IOUtils;import javax.servlet.ServletException;
 import javax.servlet.annotation.WebServlet;
 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.io.InputStream;
 import java.util.List;
 import java.util.UUID;@WebServlet("/UploadServlet")
 public class UploadServlet extends HttpServlet {
 protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
 DiskFileItemFactory factory=new DiskFileItemFactory();
 ServletFileUpload upload=new ServletFileUpload(factory);request.setCharacterEncoding("utf-8");
    //文件名中文乱码处理也可以如此写// upload.setHeaderEncoding(“utf-8”);
//设置缓冲区大小与临时文件目录
    factory.setSizeThreshold(1024*1024*10);
    File uploadTemp=new File("e:\\uploadTemp");
    uploadTemp.mkdirs();
    factory.setRepository(uploadTemp);

    //设置单个文件大小限制
    upload.setFileSizeMax(1024*1024*10);
    //设置所有文件总和大小限制
    upload.setSizeMax(1024*1024*30);

    try {
        List<FileItem> list=upload.parseRequest(request);
        System.out.println(list);
        for (FileItem fileItem:list){
            if (!fileItem.isFormField()&&fileItem.getName()!=null&&!"".equals(fileItem.getName())){
                String filName=fileItem.getName();
                //利用UUID生成伪随机字符串,作为文件名避免重复
                String uuid= UUID.randomUUID().toString();
                //获取文件后缀名
                String suffix=filName.substring(filName.lastIndexOf("."));

                //获取文件上传目录路径,在项目部署路径下的upload目录里。若想让浏览器不能直接访问到图片,可以放在WEB-INF下
                String uploadPath=request.getSession().getServletContext().getRealPath("/upload");

                File file=new File(uploadPath);
                file.mkdirs();
                //写入文件到磁盘,该行执行完毕后,若有该临时文件,将会自动删除
                fileItem.write(new File(uploadPath,uuid+suffix));
                
            }
        }
    }  catch (Exception e) {
        e.printStackTrace();
    }
}

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    doPost(request,response);
}}