Servlet3.0实现文件上传的方法

发布时间 - 2026-01-11 00:23:20    点击率:

Servlet 实现文件上传

所谓文件上传就是将本地的文件发送到服务器中保存。例如我们向百度网盘中上传本地的资源或者我们将写好的博客上传到服务器等等就是典型的文件上传。

Servlet 3.0

上次完成文件下载功能使用的是 Servlet 2.5,但是想要完成文件上传,那么继续使用 Servlet 2.5 肯定不是一个好的选择,因此我们使用 Servlet 3.0 来完成文件上传。下面我来简单介绍一下 Servlet 3.0 的新特性:

1、新增的注解支持

该版本新增了若干注解,用于简化 Servlet、过滤器(Filter)和监听器(Listener)的声明,这使得 web.xml 部署描述文件从该版本开始不再是必选的了。

2、HttpServletRequest 对文件上传的支持

此前,对于处理上传文件的操作一直是让开发者头疼的问题,因为 Servlet 本身没有对此提供直接的支持,需要使用第三方框架来实现,而且使用起来也不够简单。如今这都成为了历史,Servlet 3.0 已经提供了这个功能,而且使用也非常简单。
Servlet 3.0 的新特性当然肯定不止这些,但是其他的新特性在这里我们暂时还用不到,也就不做过多了解了。

必要条件

想要完成文件上传,肯定不是这么简单,它对浏览器端和服务器端都有许多的要求。

对浏览器的要求

  1. 一个文件的大小一般肯定不止 1 KB,既然这样,那么要上传一个文件肯定不能使用 get 方式了,所以上传文件时必须采用 post 方式。
  2. 2.表单中必须有一个文件上传项 <input type="file"> ,而且必须有 name 属性。
  3. 必须设置表单的 enctype 属性值为 multipart/form-data

对服务器的要求:

  1. 当然,我们肯定得使用 Servlet 3.0。
  2. Servlet 3.0 中接收普通上传组件(除了文件上传组件)通过 request.getParameter(String)接收,而文件上传组件通过 request.getPart(String)接收。
  3. Servlet 3.0 要求服务器必须是 Tomcat7 及其以上。

准备工作

工欲善其事,必先利其器。

1、首先,打开 Eclipse,新建一个 Dynamic Web Project

2、键入项目名,选择运行时环境为 Apache Tomcat v7.0,选择 Servlet 版本为 3.0,然后点击 Finished

3、在项目的 WebContent 目录下,新建一个文件夹 upload,用来存放上传过来的文件。

4、在 WebContent 目录下新建一个 index.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"
 pageEncoding="UTF-8"%>
<!DOCTYPE html>
<html>
 <head>
   <meta charset="UTF-8">
   <title>上传</title>
 </head>

 <body>
   <form action="${pageContext.request.contextPath}/UploadServlet" method="post" enctype="multipart/form-data">
     <label>选择一个文件:</label>
     <input type="file" name="file"><br>
     <input type="submit" value="上传"><br>
   </form>
 </body>
</html>

5、使用 Tomcat 将次项目发布,并在浏览器中预览。

将服务器启动,然后在浏览器中输入:http://localhost:8080/upload

好吧!样子有点丑,希望不要介意!如果出现以上界面,那么,准备工作就完成了!

完成案例

首先,新建一个 Servlet,在 Servlet 3.0 我们不必再为配置 web.xml 而烦恼了,只要要在 Servlet 的类名上面一行添加一个注解:

@WebServlet("/UploadServlet")

这个注解就相当与 Servlet 2.5 中的:

<servlet>
  <servlet-name>UploadServlet</servlet-name>
  <servlet-class>club.luckylight.upload.UploadServlet</servlet-class>
</servlet>
<servlet-mapping>
  <servlet-name>UploadServlet</servlet-name>
  <url-pattern>/UploadServlet</url-pattern>
</servlet-mapping>

这样比较,使用注解不是简便了很多。

然后,我们还需要添加另一个注解:

@MultipartConfig

该注解主要是为了辅助 Servlet 3.0 中 HttpServletRequest 提供的对上传文件的支持。该注解标注在 Servlet 上面,以表示该 Servlet 希望处理的请求的 MIME类型 是 multipart/form-data

接下来,我们就需要根据上传组件的 name 属性获取它了。这里我们使用 Path request.getPart(String) 方法。

Part part = request.getPart("file");

然后,我们就需要根据 part 获取头信息,然后根据头信息获取文件的路径。

在浏览器抓包,获取头信息为:

据此,我们可以获取文件名或者文件路径。

String header = part.getHeader("content-disposition");
String path = header.substring(header.indexOf("filename=") + 10, header.length() - 1);

由于获取的有可能是文件名,也有可能是文件路径,为此,有必要编写一个工具类,用来获取文件的真实名称。

/**
 * 根据文件的路径获取文件真实名称
 * 
 * @param path
 *      文件的路径
 * @return 文件名称
 */
public static String getRealName(String path) {
  int index = path.lastIndexOf("\\");

  if (index == -1) {
    index = path.lastIndexOf("/");
  }

  return path.substring(index + 1);
}

然后,调用这个方法,获得文件名。

String name = UploadUtils.getRealName(path);

接下来,我们有必要,给每个文件分配一个存放目录,因此我又编写了一个方法,用来生成一个目录。

/**
 * 根据文件名返回一个目录
 * 
 * @param name
 *      文件名称
 * @return 目录
 */
public static String getDir(String name) {
  int i = name.hashCode();
  String hex = Integer.toHexString(i);
  int j = hex.length();

  for (int k = 0; k < 8 - j; k++) {
    hex = "0" + hex;
  }

  return "/" + hex.charAt(0) + "/" + hex.charAt(1);
}

到此,万事俱备,只欠东风。我们只需要将文件拷贝到服务器。

// 获取文件的真实路径
String realPath = this.getServletContext().getRealPath("/upload" + dir);
File file = new File(realPath);

if (!file.exists()) {
  file.mkdirs();
}

// 获取输入流
InputStream inputStream = part.getInputStream();
// 定义输出流
FileOutputStream outputStream = new FileOutputStream(new File(file, name));

// 从输入流中读入数据并写到输出字节流中
int len = -1;
byte[] bytes = new byte[1024];
while ((len = inputStream.read(bytes)) != -1) {
  outputStream.write(bytes, 0, len);
}

// 关闭资源
outputStream.close();
inputStream.close();

// 删除临时文件
part.delete();

下面来测试一下:

然后,在 Tomcat webapps -> 项目名 -> upload 中就可以找到上传成功的文件了!

最后,我们打开音乐来试验下是否真的上传成功了?

嗯!薛之谦低沉的声音从耳机中传来,看来确实是上传成功了!

完整代码

UploadServlet.java

package club.luckylight.upload;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;

import javax.servlet.ServletException;
import javax.servlet.annotation.MultipartConfig;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.Part;

import club.luckylight.util.UploadUtils;

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

  private static final long serialVersionUID = 5661013723204858883L;

  protected void doGet(HttpServletRequest request, HttpServletResponse response)
      throws ServletException, IOException {
    // 获取文件上传组件
    Part part = request.getPart("file");

    // 获取文件的路径
    String header = part.getHeader("content-disposition");
    String path = header.substring(header.indexOf("filename=") + 10, header.length() - 1);

    // 获取文件名
    String name = UploadUtils.getRealName(path);

    // 获取文件的存放目录
    String dir = UploadUtils.getDir(name);

    String realPath = this.getServletContext().getRealPath("/upload" + dir);
    File file = new File(realPath);
    if (!file.exists()) {
      file.mkdirs();
    }

    // 对拷流
    InputStream inputStream = part.getInputStream();
    FileOutputStream outputStream = new FileOutputStream(new File(file, name));
    int len = -1;
    byte[] bytes = new byte[1024];
    while ((len = inputStream.read(bytes)) != -1) {
      outputStream.write(bytes, 0, len);
    }

    // 关闭资源
    outputStream.close();
    inputStream.close();

    // 删除临时文件
    part.delete();

    response.setContentType("text/html;charset=utf-8");
    response.getWriter().print("文件" + name + "上传成功!");
  }

  protected void doPost(HttpServletRequest request, HttpServletResponse response)
      throws ServletException, IOException {
    doGet(request, response);
  }

}

UploadUtils.java

package club.luckylight.util;

public class UploadUtils {

  /**
   * 根据文件的路径获取文件真实名称
   * 
   * @param path
   *      文件的路径
   * @return 文件名称
   */
  public static String getRealName(String path) {
    int index = path.lastIndexOf("\\");

    if (index == -1) {
      index = path.lastIndexOf("/");
    }

    return path.substring(index + 1);
  }

  /**
   * 根据文件名返回一个目录
   * 
   * @param name
   *      文件名称
   * @return 目录
   */
  public static String getDir(String name) {
    int i = name.hashCode();
    String hex = Integer.toHexString(i);
    int j = hex.length();

    for (int k = 0; k < 8 - j; k++) {
      hex = "0" + hex;
    }

    return "/" + hex.charAt(0) + "/" + hex.charAt(1);
  }
}

总结

这样,文件上传案例就完成了,希望对大家的学习有所帮助,也希望大家多多支持。


# servlet3.0文件上传  # servlet实现上传  # servlet文件上传下载  # Servlet3.0学习总结之基于Servlet3.0的文件上传实例  # 详解Servlet3.0新特性(从注解配置到websocket编程)  # SpringMVC + servlet3.0 文件上传的配置和实现代码  # Java Servlet3.0异步处理问题  # 文件上传  # 上传  # 新建一个  # 上传文件  # 新特性  # 器中  # 有必要  # 准备工作  # 表单  # 的是  # 临时文件  # 完成了  # 也不  # 都有  # 在这里  # 也有  # 工欲善其事  # 必先利其器  # 只欠东风  # 目录下 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何快速完成中国万网建站详细流程?  linux top下的 minerd 木马清除方法  Swift中swift中的switch 语句  laravel怎么配置Redis作为缓存驱动_laravel Redis缓存配置教程  今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  如何在IIS服务器上快速部署高效网站?  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  简单实现Android文件上传  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  如何在Windows虚拟主机上快速搭建网站?  国美网站制作流程,国美电器蒸汽鍋怎么用官方网站?  北京专业网站制作设计师招聘,北京白云观官方网站?  Laravel怎么实现搜索高亮功能_Laravel结合Scout与Algolia全文检索【实战】  Laravel如何实现一对一模型关联?(Eloquent示例)  如何快速搭建安全的FTP站点?  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  如何用搬瓦工VPS快速搭建个人网站?  东莞市网站制作公司有哪些,东莞找工作用什么网站好?  创业网站制作流程,创业网站可靠吗?  Laravel如何创建自定义Artisan命令?(代码示例)  网站制作壁纸教程视频,电脑壁纸网站?  Windows10怎样连接蓝牙设备_Windows10蓝牙连接步骤【教程】  jQuery validate插件功能与用法详解  浅述节点的创建及常见功能的实现  如何用VPS主机快速搭建个人网站?  Laravel定时任务怎么设置_Laravel Crontab调度器配置  百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏  Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】  动图在线制作网站有哪些,滑动动图图集怎么做?  如何彻底删除建站之星生成的Banner?  Claude怎样写约束型提示词_Claude约束提示词写法【教程】  三星网站视频制作教程下载,三星w23网页如何全屏?  详解Oracle修改字段类型方法总结  laravel怎么为API路由添加签名中间件保护_laravel API路由签名中间件保护方法  Laravel如何使用Socialite实现第三方登录?(微信/GitHub示例)  谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程  Laravel如何升级到最新版本?(升级指南和步骤)  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  Laravel N+1查询问题如何解决_Eloquent预加载(Eager Loading)优化数据库查询  微信h5制作网站有哪些,免费微信H5页面制作工具?  Laravel如何使用模型观察者?(Observer代码示例)  做企业网站制作流程,企业网站制作基本流程有哪些?  ,怎么在广州志愿者网站注册?  北京的网站制作公司有哪些,哪个视频网站最好?  Zeus浏览器网页版官网入口 宙斯浏览器官网在线通道  UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】  Linux网络带宽限制_tc配置实践解析【教程】  浅谈javascript alert和confirm的美化  利用 Google AI 进行 YouTube 视频 SEO 描述优化  百度浏览器如何管理插件 百度浏览器插件管理方法