如何使用 JSP/Servlet 将文件上传到服务器?
我试过这个:
<form action="upload" method="post">
<input type="text" name="description" />
<input type="file" name="file" />
<input type="submit" />
</form>
但是,我只得到文件名,没有得到文件内容。当我添加 enctype="multipart/form-data"
null
<form>
, - , - ,然后 request.getParameter()
在研究过程中,我偶然发现了 Apache Common FileUpload 。我试过这个:
FileItemFactory factory = new DiskFileItemFactory();
ServletFileUpload upload = new ServletFileUpload(factory);
List items = upload.parseRequest(request); // This line is where it died.
不幸的是,servlet 抛出了一个没有明确消息和原因的异常。这是堆栈跟踪:
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)
原文由 Thang Pham 发布,翻译遵循 CC BY-SA 4.0 许可协议
介绍
要浏览并选择要上传的文件,您需要在表单中添加 HTML
<input type="file">
字段。如 HTML 规范 中所述,您必须使用POST
方法和enctype
表单的属性必须设置为"multipart/form-data"
。提交此类表单后,二进制多部分表单数据在请求正文中可用, 格式 与未设置
enctype
时不同。在 Servlet 3.0(2009 年 12 月)之前,Servlet API 本身并不支持
multipart/form-data
。它仅支持application/x-www-form-urlencoded
的默认形式 enctype。request.getParameter()
和 consorts 在使用多部分表单数据时都会返回null
。这就是著名的 Apache Commons FileUpload 出现的地方。不要手动解析它!
理论上,您可以根据
ServletRequest#getInputStream()
解析请求正文。然而,这是一项精确而乏味的工作,需要对 RFC2388 有准确的了解。您不应该尝试自己执行此操作或复制粘贴在 Internet 其他地方找到的一些本地开发的无库代码。许多在线资源在这方面都失败了,例如 roseindia.net。另请参阅 上传 pdf 文件。您应该使用数百万用户多年来使用(并隐式测试!)的真实库。这样的库已经证明了它的健壮性。当您已经使用 Servlet 3.0 或更新版本时,请使用本机 API
如果您至少使用 Servlet 3.0(Tomcat 7、Jetty 9、JBoss AS 6、GlassFish 3 等,它们自 2010 年以来就已经存在),那么您可以只使用提供的标准 API
HttpServletRequest#getPart()
来收集单个多部分表单数据项(大多数 Servlet 3.0 实现 实际上 在幕后使用 Apache Commons FileUpload!)。此外,普通表单字段可通过getParameter()
通常的方式获得。首先用
@MultipartConfig
注释你的servlet,以让它识别和支持multipart/form-data
请求,从而得到getPart()
:8–然后,实现它的
doPost()
如下:注意
Path#getFileName()
。这是关于获取文件名的 MSIE 修复。此浏览器错误地发送完整的文件路径以及名称,而不是仅发送文件名。如果您想通过
multiple="true"
上传多个文件,或具有多个输入的老式方式,
然后你可以按照下面的方式收集它们(不幸的是没有这样的方法
request.getParts("files")
):当你还没有使用 Servlet 3.1 时,手动获取提交的文件名
请注意
Part#getSubmittedFileName()
是在 Servlet 3.1 中引入的(Tomcat 8、Jetty 9、WildFly 8、GlassFish 4 等,它们自 2013 年以来就已经存在)。如果您还没有使用 Servlet 3.1(真的吗?),那么您需要一个额外的实用方法来获取提交的文件名。请注意有关获取文件名的 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
中至少需要包含以下文件:commons-fileupload.jar
commons-io.jar
您最初的尝试很可能失败了,因为您忘记了公共 IO。
这是一个启动示例,您的
doPost()
的UploadServlet
在使用 Apache Commons FileUpload 时可能看起来像这样:It’s very important that you don’t call
getParameter()
,getParameterMap()
,getParameterValues()
,getInputStream()
,getReader()
, etc事先提出相同的要求。否则 servlet 容器将读取并解析请求主体,因此 Apache Commons FileUpload 将获得一个空请求主体。另见 servletFileUpload#parseRequest(request) returns an empty list 。注意
FilenameUtils#getName()
。这是关于获取文件名的 MSIE 修复。此浏览器错误地发送完整的文件路径以及名称,而不是仅发送文件名。或者,您也可以将所有内容包装在
Filter
中,它会自动解析所有内容并将这些内容放回请求的参数图中,以便您可以继续使用request.getParameter()
通常的方式并检索request.getAttribute()
上传的文件。 您可以在这篇博客文章中找到示例。GlassFish3 错误的解决方法
getParameter()
仍然返回null
请注意,早于 3.1.2 的 Glassfish 版本有 一个错误,其中
getParameter()
仍然返回null
。如果您的目标是这样的容器并且无法升级它,那么您需要借助此实用程序方法从getPart()
中提取值:保存上传的文件(不要使用
getRealPath()
或part.write()
!)前往以下答案,了解有关将获得的
InputStream
(如上述代码片段所示的fileContent
变量)正确保存到磁盘或数据库的详细信息:服务上传的文件
有关将保存的文件从磁盘或数据库正确提供回客户端的详细信息,请参阅以下答案:
Ajax化表单
前往以下关于如何使用 Ajax(和 jQuery)上传的答案。请注意,不需要为此更改用于收集表单数据的 servlet 代码!只有您响应的方式可能会改变,但这是微不足道的(即不是转发到 JSP,而是打印一些 JSON 或 XML 甚至纯文本,具体取决于负责 Ajax 调用的脚本所期望的内容)。
希望这一切都有帮助:)