介绍
要浏览并选择要上传的文件,您需要<input type="file">
在表单中有一个HTML 字段。如HTML规范中所述,您必须使用POST
方法,并且enctype
表单的属性必须设置为"multipart/form-data"
。
<form action="upload" method="post" enctype="multipart/form-data">
<input type="text" name="description" />
<input type="file" name="file" />
<input type="submit" />
</form>
提交此类表单后,二进制多部分表单数据在请求正文中的使用方式与enctype
未设置时的格式不同。
在Servlet 3.0之前,Servlet API本身不支持multipart/form-data
。它仅支持默认格式enctype application/x-www-form-urlencoded
。使用多部分表单数据时,request.getParameter()
and和con都会全部返回null
。这就是众所周知的Apache Commons FileUpload出现的地方。
不要手动解析它!
从理论上讲,您可以根据自己解析请求主体ServletRequest#getInputStream()
。但是,这是一项精确而乏味的工作,需要对RFC2388有精确的了解。您不应该尝试自己执行此操作,也不应该复制粘贴互联网上其他地方找到的一些本地编写的无库代码。许多在线资源在此方面都失败了,例如roseindia.net。另请参阅pdf文件的上载。您应该使用一个真正的库,数百万用户使用该库多年(并进行了隐式测试!)。这样的库已经证明了其健壮性。
如果您已经在Servlet 3.0或更高版本上,请使用本机API
如果您至少使用Servlet 3.0(Tomcat 7,Jetty 9,JBoss AS 6,GlassFish 3等),则可以使用提供的标准API HttpServletRequest#getPart()
来收集各个部分的表单数据项(大多数Servlet 3.0实现实际上都使用Apache) Commons FileUpload的内容在这里!)。此外,可以通过getParameter()
常规方式使用普通表单字段。
首先用注释您的servlet,@MultipartConfig
以使其识别并支持multipart/form-data
请求,然后开始getPart()
工作:
@WebServlet("/upload")
@MultipartConfig
public class UploadServlet extends HttpServlet {
// ...
}
然后,doPost()
按以下方式实施:
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String description = request.getParameter("description"); // Retrieves <input type="text" name="description">
Part filePart = request.getPart("file"); // Retrieves <input type="file" name="file">
String fileName = Paths.get(filePart.getSubmittedFileName()).getFileName().toString(); // MSIE fix.
InputStream fileContent = filePart.getInputStream();
// ... (do your job here)
}
注意Path#getFileName()
。这是有关获取文件名的MSIE修复程序。该浏览器错误地沿名称发送了完整的文件路径,而不仅仅是文件名。
如果您要<input type="file" name="file" multiple="true" />
上传多文件,请按以下方式收集它们(不幸的是,没有这样的方法request.getParts("file")
):
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// ...
List<Part> fileParts = request.getParts().stream().filter(part -> "file".equals(part.getName()) && part.getSize() > 0).collect(Collectors.toList()); // Retrieves <input type="file" name="file" multiple="true">
for (Part filePart : fileParts) {
String fileName = Paths.get(filePart.getSubmittedFileName()).getFileName().toString(); // MSIE fix.
InputStream fileContent = filePart.getInputStream();
// ... (do your job here)
}
}
当您尚未使用Servlet 3.1时,请手动获取已提交的文件名
请注意,它Part#getSubmittedFileName()
是在Servlet 3.1中引入的(Tomcat 8,Jetty 9,WildFly 8,GlassFish 4等)。如果尚未使用Servlet 3.1,则需要其他实用程序方法来获取提交的文件名。
private static String getSubmittedFileName(Part part) {
for (String cd : part.getHeader("content-disposition").split(";")) {
if (cd.trim().startsWith("filename")) {
String fileName = cd.substring(cd.indexOf('=') + 1).trim().replace("\"", "");
return fileName.substring(fileName.lastIndexOf('/') + 1).substring(fileName.lastIndexOf('\\') + 1); // MSIE fix.
}
}
return null;
}
String fileName = getSubmittedFileName(filePart);
请注意有关获取文件名的MSIE修复程序。该浏览器错误地沿名称发送了完整的文件路径,而不仅仅是文件名。
当您尚未使用Servlet 3.0时,请使用Apache Commons FileUpload
如果您还没有使用Servlet 3.0(不是时候升级了吗?),通常的做法是利用Apache Commons FileUpload解析多部分表单数据请求。它具有出色的用户指南和常见问题解答(请仔细阅读两者)。也有O'Reilly(“ cos ”)MultipartRequest
,但是它有一些(较小的)错误,并且多年来一直没有积极维护。我不建议使用它。Apache Commons FileUpload仍在积极维护中,目前非常成熟。
为了使用Apache Commons FileUpload,您需要在Webapp的文件中至少包含以下文件/WEB-INF/lib
:
您最初的尝试很可能失败,因为您忘记了公共IO。
这里有一个例子开球怎么doPost()
你的UploadServlet
可以看看使用Apache通用FileUpload时,如:
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
try {
List<FileItem> items = new ServletFileUpload(new DiskFileItemFactory()).parseRequest(request);
for (FileItem item : items) {
if (item.isFormField()) {
// Process regular form field (input type="text|radio|checkbox|etc", select, etc).
String fieldName = item.getFieldName();
String fieldValue = item.getString();
// ... (do your job here)
} else {
// Process form file field (input type="file").
String fieldName = item.getFieldName();
String fileName = FilenameUtils.getName(item.getName());
InputStream fileContent = item.getInputStream();
// ... (do your job here)
}
}
} catch (FileUploadException e) {
throw new ServletException("Cannot parse multipart request.", e);
}
// ...
}
这是非常重要的,你不叫getParameter()
,getParameterMap()
,getParameterValues()
,getInputStream()
,getReader()
等上预先同样的要求。否则,servlet容器将读取并解析请求主体,因此Apache Commons FileUpload将获得一个空的请求主体。另请参见ServletFileUpload#parseRequest(request)返回一个空列表。
注意FilenameUtils#getName()
。这是有关获取文件名的MSIE修复程序。该浏览器错误地沿名称发送了完整的文件路径,而不仅仅是文件名。
或者,您也可以将所有内容包装Filter
在内,后者会自动对其进行解析,然后将其放回请求的参数图中,以便您可以继续使用request.getParameter()
通常的方法并通过检索上载的文件request.getAttribute()
。您可以在此博客文章中找到示例。
GlassFish3错误getParameter()
仍然存在的解决方法null
请注意,低于3.1.2的Glassfish版本存在一个错误,该错误getParameter()
仍会返回null
。如果您针对此类容器并且无法升级,则需要getPart()
借助此实用程序方法从中提取值:
private static String getValue(Part part) throws IOException {
BufferedReader reader = new BufferedReader(new InputStreamReader(part.getInputStream(), "UTF-8"));
StringBuilder value = new StringBuilder();
char[] buffer = new char[1024];
for (int length = 0; (length = reader.read(buffer)) > 0;) {
value.append(buffer, 0, length);
}
return value.toString();
}
String description = getValue(request.getPart("description")); // Retrieves <input type="text" name="description">
保存上传的文件(请勿使用getRealPath()
或part.write()
!)
前往以下答案,详细了解如何正确地将获得的InputStream
(fileContent
上面代码片段中显示的变量)保存到磁盘或数据库中:
提供上传的文件
前往以下答案,详细了解如何将已保存的文件从磁盘或数据库正确地提供给客户端:
合并表格
前往以下答案,了解如何使用Ajax(和jQuery)上传。请注意,不需要为此更改用于收集表单数据的servlet代码!只有响应的方式可能会改变,但这很简单(即,不转发到JSP,仅打印一些JSON或XML甚至纯文本,具体取决于负责Ajax调用的脚本是什么)。
希望这对您有所帮助:)