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是apache的commons组件提供的上传组件,它最主要的工作就是帮我们解析request.getInpustream()。可以参考在线API文档:http://tool.oschina.net/apidocs/apidoc?api=commons-fileupload
使用fileUpload组件首先需要引入两个jar包:
fileUpload的核心类有DiskFileItemFactory、ServletFileUpload、FileItem。
使用fileUpload固定步骤:
一个FileItem对象对应一个表单项。FileItem类有如下方法:
表单提交页面同上,修改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
濂藉ソ瀛︿範 澶╁ぉ鍚戜笂
*************
使用fileUpload组件实现文件上传除了上面的那些方法之外还要注意的:
注:如果没有指定临时文件目录,默认采用系统默认的临时文件路径,可以通过System.getProperty("java.io.tmpdir")获取,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 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);
}
}
参考博客:https://blog.csdn.net/fsdadsadas/article/details/73162677
https://www.cnblogs.com/lxboy2009/p/5994743.html