本示例演示如何使用 Java Servlet 上传文件

始终可以将文件上传到 Java EE Servlet,但是要完成该工作需要付出很大的努力。 Apache Foundation 甚至构建了一个称为 Commons FileUpload 的库,以使此任务更易于实现。 尽管如此,Servlet 3.0 规范弥补了这一差距,并且自 Java EE 6 起,多部分配置选项已添加到 Servlet 中,从而在HttpServletRequest中引入了getPartgetParts方法。

如果您有兴趣使用 WebServices 将文件上传到服务器,则可以查看本教程

Servlet 文件上传示例

Servlet 文件上载示例演示了MultipartConfig注解的用法,并使用户能够上载一个或两个文件。

该示例项目的结构非常简单。 它由一个 servlet 文件FileUploadServlet.javapom.xml和可选的web.xml组成,这些文件用于在构建时处理依赖项。 正如我们在 Servlet 注释示例中讨论的那样,您可以在注释和部署描述符之间进行选择,以设置 Servlet 配置。 本示例使用注释。

下图显示了项目结构

Java Servlet 文件上传 - 图1

Servlet 文件上传项目结构

在 Maven 的pom.xml文件中,我们需要声明的唯一依赖项是javax.servlet

  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  2. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  3. <modelVersion>4.0.0</modelVersion>
  4. <groupId>net.javatutorial.tutorials</groupId>
  5. <artifactId>ServletFileUpload</artifactId>
  6. <version>1</version>
  7. <packaging>war</packaging>
  8. <name>ServletFileUpload</name>
  9. <url>https://javatutorial.net</url>
  10. <properties>
  11. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  12. </properties>
  13. <dependencies>
  14. <dependency>
  15. <groupId>javax.servlet</groupId>
  16. <artifactId>javax.servlet-api</artifactId>
  17. <version>3.1.0</version>
  18. <scope>provided</scope>
  19. </dependency>
  20. </dependencies>
  21. <build>
  22. <finalName>fileuploader</finalName>
  23. <sourceDirectory>src/main/java</sourceDirectory>
  24. <plugins>
  25. <plugin>
  26. <groupId>org.apache.maven.plugins</groupId>
  27. <artifactId>maven-war-plugin</artifactId>
  28. <version>2.3</version>
  29. <configuration>
  30. <warSourceDirectory>src/main/webapp</warSourceDirectory>
  31. </configuration>
  32. </plugin>
  33. <plugin>
  34. <groupId>org.apache.maven.plugins</groupId>
  35. <artifactId>maven-compiler-plugin</artifactId>
  36. <version>3.1</version>
  37. <configuration>
  38. <source>1.8</source>
  39. <target>1.8</target>
  40. </configuration>
  41. </plugin>
  42. </plugins>
  43. </build>
  44. </project>

文件上传 servlet 是我们项目的核心。 它只有两种方法 - goGet(显示上载表格)和doPost(完成整个上载工作)。

  1. package net.javatutorial.tutorials;
  2. import java.io.File;
  3. import java.io.IOException;
  4. import java.io.PrintWriter;
  5. import javax.servlet.ServletException;
  6. import javax.servlet.annotation.MultipartConfig;
  7. import javax.servlet.annotation.WebServlet;
  8. import javax.servlet.http.HttpServlet;
  9. import javax.servlet.http.HttpServletRequest;
  10. import javax.servlet.http.HttpServletResponse;
  11. import javax.servlet.http.Part;
  12. @WebServlet(name = "uploadServlet", urlPatterns = { "/upload" }, loadOnStartup = 1)
  13. @MultipartConfig(fileSizeThreshold = 6291456, // 6 MB
  14. maxFileSize = 10485760L, // 10 MB
  15. maxRequestSize = 20971520L // 20 MB
  16. )
  17. public class FileUploadServlet extends HttpServlet {
  18. private static final long serialVersionUID = 5619951677845873534L;
  19. private static final String UPLOAD_DIR = "uploads";
  20. @Override
  21. protected void doGet(HttpServletRequest request, HttpServletResponse response)
  22. throws ServletException, IOException {
  23. response.setContentType("text/html");
  24. response.setCharacterEncoding("UTF-8");
  25. PrintWriter writer = response.getWriter();
  26. writer.append("<!DOCTYPE html>\r\n")
  27. .append("<html>\r\n")
  28. .append(" <head>\r\n")
  29. .append(" <title>File Upload Form</title>\r\n")
  30. .append(" </head>\r\n")
  31. .append(" <body>\r\n");
  32. writer.append("<h1>Upload file</h1>\r\n");
  33. writer.append("<form method=\"POST\" action=\"upload\" ")
  34. .append("enctype=\"multipart/form-data\">\r\n");
  35. writer.append("<input type=\"file\" name=\"fileName1\"/><br/><br/>\r\n");
  36. writer.append("<input type=\"file\" name=\"fileName2\"/><br/><br/>\r\n");
  37. writer.append("<input type=\"submit\" value=\"Submit\"/>\r\n");
  38. writer.append("</form>\r\n");
  39. writer.append(" </body>\r\n").append("</html>\r\n");
  40. }
  41. @Override
  42. protected void doPost(HttpServletRequest request, HttpServletResponse response)
  43. throws ServletException, IOException {
  44. response.setContentType("text/html");
  45. response.setCharacterEncoding("UTF-8");
  46. // gets absolute path of the web application
  47. String applicationPath = request.getServletContext().getRealPath("");
  48. // constructs path of the directory to save uploaded file
  49. String uploadFilePath = applicationPath + File.separator + UPLOAD_DIR;
  50. // creates upload folder if it does not exists
  51. File uploadFolder = new File(uploadFilePath);
  52. if (!uploadFolder.exists()) {
  53. uploadFolder.mkdirs();
  54. }
  55. PrintWriter writer = response.getWriter();
  56. // write all files in upload folder
  57. for (Part part : request.getParts()) {
  58. if (part != null && part.getSize() > 0) {
  59. String fileName = part.getSubmittedFileName();
  60. String contentType = part.getContentType();
  61. // allows only JPEG files to be uploaded
  62. if (!contentType.equalsIgnoreCase("image/jpeg")) {
  63. continue;
  64. }
  65. part.write(uploadFilePath + File.separator + fileName);
  66. writer.append("File successfully uploaded to "
  67. + uploadFolder.getAbsolutePath()
  68. + File.separator
  69. + fileName
  70. + "<br>\r\n");
  71. }
  72. }
  73. }
  74. }

@MultipatrtConfig注解使 Servlet 可以接受文件上传。 有 3 个重要属性:

  • fileSizeThreshold – 在将文件写入temp目录之前要超出的文件大小。 如果文件小于此阈值,则文件将在请求完成之前驻留在内存中。
  • maxFileSize – 这是允许上传的文件的最大大小。 在上面的示例中,不允许用户上传大于 10 MB 的文件
  • maxRequestSize – 是我们尝试通过一个请求上传的所有文件的大小的总和。 在上面的示例中,我们将该值设置为 20 MB,这意味着无论文件数量多少,我们总共可以上传 20MB

您可能要指定或不指定第四个属性。 它称为location,它指向 Web 容器应存储临时文件的目录。 但是,如果您未指定此属性,则容器将使用默认的temp文件夹。

我们重写doGet方法以显示具有两个文件选择器字段的简单形式。 您可能需要添加其他输入字段,因为多部分附件允许这样做。

Java Servlet 文件上传 - 图2

表格上传文件

doPost方法中,我们首先构造要存储上载文件的文件夹的路径。 比我们使用request.getParts()遍历用户选择上传的文件,最后将它们存储到所需位置。

构建并部署后,您可以在以下浏览器中访问应用程序:http://localhost:8080/fileuploader/upload

您可以在我们的 GitHub 存储库的中找到该项目