如何使用JSP / Servlet将文件上传到服务器?

问题:

如何使用JSP / Servlet将文件上传到服务器?我试过这个:
class=”lang-html prettyprint-override”>

<form action="upload" method="post">
    <input type="text" name="description" />
    <input type="file" name="file" />
    <input type="submit" />
</form>

但是,我只获取文件名,而不是文件内容。当enctype="multipart/form-data"添加到<form>时,request.getParameter()返回null
在研究期间,我偶然发现Apache Common FileUpload。我试过这个:

FileItemFactory factory = new DiskFileItemFactory();
ServletFileUpload upload = new ServletFileUpload(factory);
List items = upload.parseRequest(request); // This line is where it died.

不幸的是,servlet抛出一个异常,没有明确的消息和原因。这是stacktrace:

SEVERE: Servlet.service() for servlet UploadServlet threw exception
javax.servlet.ServletException: Servlet execution threw an exception
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:313)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:102)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:298)
    at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:852)
    at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:588)
    at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489)
    at java.lang.Thread.run(Thread.java:637)

回答:

介绍

要浏览并选择要上传的文件,您需要在表单中的HTML <input type="file">字段。如HTML specification所述,您必须使用POST方法,并且必须将表单的enctype属性设置为"multipart/form-data"
class=”lang-html prettyprint-override”>

<form action="upload" method="post" enctype="multipart/form-data">
    <input type="text" name="description" />
    <input type="file" name="file" />
    <input type="submit" />
</form>

提交这样的表单后,二进制多部分表单数据在a different format中的请求正文中比未设置enctype时可用。
在Servlet 3.0之前,Servlet API并没有支持multipart/form-data。它仅支持application/x-www-form-urlencoded的默认形式enctype。使用多部分表单数据时,request.getParameter()和consorts都将返回null。这是众所周知的Apache Commons FileUpload出现的地方。

不要手动解析它!

您可以在理论上根据ServletRequest#getInputStream()自己解析请求体。然而,这是一个精确和乏味的工作,需要精确的RFC2388知识。你不应该尝试这样做,或者是在互联网上的其他地方找到一些本土化的免库存代码。许多在线资源在这方面失败了,比如roseindia.net。另见uploading of pdf file。您应该使用数百万用户多年使用(并隐含测试!)的真实库。这样的图书馆已经证明了它的坚固性。

当您已经在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()获得正常表单字段。
首先使用@MultipartConfig注释您的servlet,以便让其识别并支持multipart/form-data请求,从而使getPart()工作:

@WebServlet("/upload")
@MultipartConfig
public class UploadServlet extends HttpServlet {
    // ...
}

然后,按如下方式实现其doPost()
class=”lang-java prettyprint-override”>

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())).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时,手动获取提交的文件名

请注意,Servlet 3.1(Tomcat 8,Jetty 9,WildFly 8,GlassFish 4等)中引入了Part#getSubmittedFileName()。如果您尚未使用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来解析多部分表单数据请求。它有一个很好的User GuideFAQ(仔细地通过两者)。还有O’Reilly(“cos”)MultipartRequest,但它有一些(次要的)错误,并且几年来没有被积极维护。我不会建议使用它。 Apache Commons FileUpload仍然积极维护,目前非常成熟。
为了使用Apache Commons FileUpload,您需要在webapp的/WEB-INF/lib中至少包含以下文件:

您的初始尝试失败最有可能是因为您忘记了公共IO。
以下是一个开始示例,当您使用Apache Commons FileUpload时,您UploadServletdoPost()可能如此:
class=”lang-java prettyprint-override”>

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将获得一个空请求体。另见a.o. ServletFileUpload#parseRequest(request) returns an empty list
注意FilenameUtils#getName()。这是获取文件名的MSIE修复。该浏览器不正确地发送完整的文件路径,而不是只有文件名。
或者,您也可以将所有内容全部包含在Filter中,全部自动解析,并将该信息放回请求的参数中,以便您可以通过request.getParameter()继续使用request.getParameter()并通过request.getAttribute()检索上传的文件。 You can find an example in this blog article

GlassFish3错误的解决方法

请注意,超过3.1.2的Glassfish版本为a bug,其中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">

保存上传的文件(不要使用

要获得正确保存获取的InputStream(上述代码片段所示的fileContent变量)到磁盘或数据库的详细信息,请参阅以下答案:

提供上传的文件

有关正确将保存的文件从磁盘或数据库提供给客户端的详细信息,请参阅以下答案:

Ajaxifying表单

转到以下答案如何使用Ajax(和jQuery)进行上传。请注意,收集表单数据的servlet代码不需要更改!只有你的回应方式可能会改变,但这是相当微不足道的(而不是转发到JSP,只需打印一些JSON或XML,甚至是纯文本,这取决于Ajax调用所负责的任何脚本)。

希望这一切有所帮助:)

 
 
Code问答: http://codewenda.com/topics/python/
Stackoverflow: How to upload files to server using JSP/Servlet?

*转载请注明本文链接以及stackoverflow的英文链接

发表评论

电子邮件地址不会被公开。 必填项已用*标注

3 + 2 =