Servlet 文件上传


Servlet 可以与 HTML form 标签一起使用,来允许用户上传文件到服务器。上传的文件可以是文本文件或图像文件或任何文档。

创建一个文件上传表单

下面的 HTML 代码创建了一个文件上传表单。以下几点需要注意:

  • 表单 method 属性应该设置为 POST 方法,不能使用 GET 方法。
  • 表单 enctype 属性应该设置为 multipart/form-data.
  • 表单 action 属性应该设置为在后端服务器上处理文件上传的 Servlet 文件。下面的实例使用了 UploadServlet Servlet 来上传文件。
  • 上传单个文件,您应该使用单个带有属性 type="file" 的 <input .../> 标签。为了允许多个文件上传,请包含多个 name 属性值不同的 input 标签。输入标签具有不同的名称属性的值。浏览器会为每个 input 标签关联一个浏览按钮。
  1. <html>
  2. <head>
  3. <title>文件上传表单</title>
  4. </head>
  5. <body>
  6. <h3>文件上传:</h3>
  7. 请选择要上传的文件:<br />
  8. <form action="UploadServlet" method="post"
  9. enctype="multipart/form-data">
  10. <input type="file" name="file" size="50" />
  11. <br />
  12. <input type="submit" value="上传文件" />
  13. </form>
  14. </body>
  15. </html>

这将显示下面的结果,允许用户从本地计算机选择一个文件,当用户点击"上传文件"时,表单会连同从本地计算机选择的文件一起提交:

  1. <b>文件上传:</b>
  2. 请选择要上传的文件:<br />
  3. <input type="file" name="file" size="50" />
  4. <br />
  5. <input type="button" value="上传文件" />
  6. <br />
  7. 注:这只是虚拟的表单,不会正常工作。

编写后台 Servlet

以下是 Servlet UploadServlet,会接受上传的文件,并把它储存在目录 <Tomcat-installation-directory>/webapps/data 中。这个目录名也可以使用外部配置来添加,比如 web.xml 中的 context-param 元素,如下所示:

  1. <web-app>
  2. ....
  3. <context-param>
  4. <description>Location to store uploaded file</description>
  5. <param-name>file-upload</param-name>
  6. <param-value>
  7. c:\apache-tomcat-5.5.29\webapps\data\
  8. </param-value>
  9. </context-param>
  10. ....
  11. </web-app>

以下是 UploadServlet 的源代码,可以一次处理多个文件的上传。在继续操作之前,请确认下列各项:

  • 下面的实例依赖于 FileUpload,所以一定要确保在您的 classpath 中有最新版本的 commons-fileupload.x.x.jar 文件。可以从 http://commons.apache.org/fileupload/ 下载。
  • FileUpload 依赖于 Commons IO,所以一定要确保在您的 classpath 中有最新版本的 commons-io-x.x.jar 文件。可以从 http://commons.apache.org/io/ 下载。
  • 在测试下面实例时,您上传的文件大小不能大于 maxFileSize,否则文件将无法上传。
  • 请确保已经提前创建好目录 c:\temp and c:\apache-tomcat-5.5.29\webapps\data。
  1. // 导入必需的 java 库
  2. import java.io.*;
  3. import java.util.*;
  4. import javax.servlet.ServletConfig;
  5. import javax.servlet.ServletException;
  6. import javax.servlet.http.HttpServlet;
  7. import javax.servlet.http.HttpServletRequest;
  8. import javax.servlet.http.HttpServletResponse;
  9. import org.apache.commons.fileupload.FileItem;
  10. import org.apache.commons.fileupload.FileUploadException;
  11. import org.apache.commons.fileupload.disk.DiskFileItemFactory;
  12. import org.apache.commons.fileupload.servlet.ServletFileUpload;
  13. import org.apache.commons.io.output.*;
  14.  
  15. public class UploadServlet extends HttpServlet {
  16. private boolean isMultipart;
  17. private String filePath;
  18. private int maxFileSize = 50 * 1024;
  19. private int maxMemSize = 4 * 1024;
  20. private File file ;
  21.  
  22. public void init( ){
  23. // 获取文件将被存储的位置
  24. filePath =
  25. getServletContext().getInitParameter("file-upload");
  26. }
  27. public void doPost(HttpServletRequest request,
  28. HttpServletResponse response)
  29. throws ServletException, java.io.IOException {
  30. // 检查我们有一个文件上传请求
  31. isMultipart = ServletFileUpload.isMultipartContent(request);
  32. response.setContentType("text/html");
  33. java.io.PrintWriter out = response.getWriter( );
  34. if( !isMultipart ){
  35. out.println("<html>");
  36. out.println("<head>");
  37. out.println("<title>Servlet upload</title>");
  38. out.println("</head>");
  39. out.println("<body>");
  40. out.println("<p>No file uploaded</p>");
  41. out.println("</body>");
  42. out.println("</html>");
  43. return;
  44. }
  45. DiskFileItemFactory factory = new DiskFileItemFactory();
  46. // 文件大小的最大值将被存储在内存中
  47. factory.setSizeThreshold(maxMemSize);
  48. // Location to save data that is larger than maxMemSize.
  49. factory.setRepository(new File("c:\\temp"));
  50.  
  51. // 创建一个新的文件上传处理程序
  52. ServletFileUpload upload = new ServletFileUpload(factory);
  53. // 允许上传的文件大小的最大值
  54. upload.setSizeMax( maxFileSize );
  55.  
  56. try{
  57. // 解析请求,获取文件项
  58. List fileItems = upload.parseRequest(request);
  59. // 处理上传的文件项
  60. Iterator i = fileItems.iterator();
  61.  
  62. out.println("<html>");
  63. out.println("<head>");
  64. out.println("<title>Servlet upload</title>");
  65. out.println("</head>");
  66. out.println("<body>");
  67. while ( i.hasNext () )
  68. {
  69. FileItem fi = (FileItem)i.next();
  70. if ( !fi.isFormField () )
  71. {
  72. // 获取上传文件的参数
  73. String fieldName = fi.getFieldName();
  74. String fileName = fi.getName();
  75. String contentType = fi.getContentType();
  76. boolean isInMemory = fi.isInMemory();
  77. long sizeInBytes = fi.getSize();
  78. // 写入文件
  79. if( fileName.lastIndexOf("\\") >= 0 ){
  80. file = new File( filePath +
  81. fileName.substring( fileName.lastIndexOf("\\"))) ;
  82. }else{
  83. file = new File( filePath +
  84. fileName.substring(fileName.lastIndexOf("\\")+1)) ;
  85. }
  86. fi.write( file ) ;
  87. out.println("Uploaded Filename: " + fileName + "<br>");
  88. }
  89. }
  90. out.println("</body>");
  91. out.println("</html>");
  92. }catch(Exception ex) {
  93. System.out.println(ex);
  94. }
  95. }
  96. public void doGet(HttpServletRequest request,
  97. HttpServletResponse response)
  98. throws ServletException, java.io.IOException {
  99. throw new ServletException("GET method used with " +
  100. getClass( ).getName( )+": POST method required.");
  101. }
  102. }

编译和运行 Servlet

编译上面的 Servlet UploadServlet,并在 web.xml 文件中创建所需的条目,如下所示:

  1. <servlet>
  2. <servlet-name>UploadServlet</servlet-name>
  3. <servlet-class>UploadServlet</servlet-class>
  4. </servlet>
  5.  
  6. <servlet-mapping>
  7. <servlet-name>UploadServlet</servlet-name>
  8. <url-pattern>/UploadServlet</url-pattern>
  9. </servlet-mapping>

现在尝试使用您在上面创建的 HTML 表单来上传文件。当您在浏览器中访问:http://localhost:8080/UploadFile.htm 时,它会显示下面的结果,这将有助于您从本地计算机上传任何文件。

  1. <b>文件上传:</b>
  2. 请选择要上传的文件:<br />
  3. <input type="file" name="file" size="50" />
  4. <br />
  5. <input type="button" value="上传文件" />

如果您的 Servelt 脚本能正常工作,那么您的文件会被上传到 c:\apache-tomcat-5.5.29\webapps\data\ 目录中。