从 01 开始 从 01 开始
首页
  • 📚 计算机基础

    • 计算机简史
    • 数字电路
    • 计算机组成原理
    • 操作系统
    • Linux
    • 计算机网络
    • 数据库
    • 编程工具
    • 装机
  • 🎨 前端

    • Node
  • JavaSE
  • Java 高级
  • JavaEE

    • 构建、依赖管理
    • Ant
    • Maven
    • 日志框架
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • 环境管理和配置管理-科普篇
    • Servlet
  • Spring

    • Spring基础
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC

    • SpringMVC 基础
  • SpringBoot

    • SpringBoot 基础
  • Windows 使用技巧
  • 手机
  • 最全面的输入法教程
  • 最全面的浏览器教程
  • Office
  • 图片类工具
  • 效率类工具
  • RSS
  • 码字工具
  • 各大平台
  • 校招
  • 五险一金等
  • 职场规划
  • 关于离职
  • 杂谈
  • 📖 读书

    • 读书工具
    • 读书笔记
  • 🌍 英语

    • 从零开始学英语
    • 英语兔的相关视频
    • Larry 想做技术大佬的相关视频
  • 🏛️ 政治

    • 反腐
    • GFW
    • 404 内容
    • 审查与自我审查
    • 互联网
    • 战争
  • 💰 经济

    • 关于税
    • 理财
  • 💪 健身

    • 睡眠
    • 皮肤
    • 口腔健康
    • 学会呼吸
    • 健身日志
  • 🏠 其他

    • 驾驶技能
    • 租房与买房
    • 厨艺
  • 电影

    • 电影推荐
  • 电视剧
  • 漫画

    • 漫画软件
    • 漫画推荐
  • 游戏

    • Steam
    • 三国杀
    • 求生之路
  • 小说
  • 关于本站
  • 关于博主
  • 打赏
  • 网站动态
  • 友人帐
  • 从零开始搭建博客
  • 搭建邮件服务器
  • 本站分享
  • 🌈 生活

    • 2022
    • 2023
    • 2024
    • 2025
  • 📇 文章索引

    • 文章分类
    • 文章归档

晓林

程序猿,自由职业者,博主,英语爱好者,健身达人
首页
  • 📚 计算机基础

    • 计算机简史
    • 数字电路
    • 计算机组成原理
    • 操作系统
    • Linux
    • 计算机网络
    • 数据库
    • 编程工具
    • 装机
  • 🎨 前端

    • Node
  • JavaSE
  • Java 高级
  • JavaEE

    • 构建、依赖管理
    • Ant
    • Maven
    • 日志框架
    • Junit
    • JDBC
    • XML-JSON
  • JavaWeb

    • 服务器软件
    • 环境管理和配置管理-科普篇
    • Servlet
  • Spring

    • Spring基础
  • 主流框架

    • Redis
    • Mybatis
    • Lucene
    • Elasticsearch
    • RabbitMQ
    • MyCat
    • Lombok
  • SpringMVC

    • SpringMVC 基础
  • SpringBoot

    • SpringBoot 基础
  • Windows 使用技巧
  • 手机
  • 最全面的输入法教程
  • 最全面的浏览器教程
  • Office
  • 图片类工具
  • 效率类工具
  • RSS
  • 码字工具
  • 各大平台
  • 校招
  • 五险一金等
  • 职场规划
  • 关于离职
  • 杂谈
  • 📖 读书

    • 读书工具
    • 读书笔记
  • 🌍 英语

    • 从零开始学英语
    • 英语兔的相关视频
    • Larry 想做技术大佬的相关视频
  • 🏛️ 政治

    • 反腐
    • GFW
    • 404 内容
    • 审查与自我审查
    • 互联网
    • 战争
  • 💰 经济

    • 关于税
    • 理财
  • 💪 健身

    • 睡眠
    • 皮肤
    • 口腔健康
    • 学会呼吸
    • 健身日志
  • 🏠 其他

    • 驾驶技能
    • 租房与买房
    • 厨艺
  • 电影

    • 电影推荐
  • 电视剧
  • 漫画

    • 漫画软件
    • 漫画推荐
  • 游戏

    • Steam
    • 三国杀
    • 求生之路
  • 小说
  • 关于本站
  • 关于博主
  • 打赏
  • 网站动态
  • 友人帐
  • 从零开始搭建博客
  • 搭建邮件服务器
  • 本站分享
  • 🌈 生活

    • 2022
    • 2023
    • 2024
    • 2025
  • 📇 文章索引

    • 文章分类
    • 文章归档
  • JavaSE

  • JavaSenior

  • JavaEE

  • JavaWeb

    • 服务器软件

    • 环境管理和配置管理-科普篇
    • Servlet入门

      • 什么是 Servlet
      • Servlet 入门案例
      • Servlet 生命周期
      • Servlet 中的注解
      • Tomcat 集成 IDEA
      • Servlet 体系结构
      • HTTP 协议基础
      • 深入 request 和 response 对象
      • request 对象基本使用
      • request 其他功能
      • Servlet 实现登录功能
      • HTTP 协议基础-响应
      • Response 对象基本使用
      • response 对象之重定向
      • response 输出字符到浏览器
      • response 输出字节数据
      • 验证码案例
      • ServletContext
      • 文件下载案例
        • 需求
        • 下载页面
        • 下载 Servlet
        • 如果文件名包含中文名
        • 总结
      • Cookie 笔记
      • Cookie 的更多细节
      • Cookie 实践:记住上次访问时间
      • JSP 入门
      • JSP 的内置对象和案例
      • IDEA 与 JavaWeb 的小技巧
      • Session 笔记
      • 验证码案例
      • JSP 深入学习
      • MVC 开发模式
      • EL 表达式和 JSTL 标签
      • JSTL 标签库
      • 案例:列表的增删改查
      • Filter 学习
      • Filter 案例
      • Listener 学习
      • Java 中的 Ajax
      • Java 中的 JSON
  • Spring

  • 主流框架

  • SpringMVC

  • SpringBoot

  • Java
  • JavaWeb
  • Servlet入门
2023-04-17
目录

文件下载案例

# 75.文件下载案例

我们来做一个下载文件的功能,加深我们对 Servlet 的使用。

‍

# 需求

  1. 页面显示超链接
  2. 点击超链接后弹出下载提示框
  3. 完成图片文件下载

# 下载页面

新建一个 download.html

在 web 目录下新建 img 目录,并放两个图片 1.jpg 2.jpg

如果超链接直接指向图片资源的路径,那么由于该图片能被浏览器显示,浏览器会直接展示,而不是弹出下载提示框:

<!DOCTYPE html>
<html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>文件下载案例</title>
    </head>
    <body>
      <a href="/hello/img/1.jpg">图片1</a>
    </body>
</html>
1
2
3
4
5
6
7
8
9
10

‍ 访问 http://localhost: 8080/hello/download.html,点击超链接,浏览器会直接展示图片:

而如果超链接指向的文件,不能被浏览器直接显示,才会提示下载框,例如我们放一个视频文件 1.mp4

 <a href="/hello/img/1.jpg">图片1</a> <br>
<a href="/hello/img/1.mp4">视频1</a> <br>
1
2

‍ ‍

解决方法:我们设置响应头,告诉浏览器资源的打开方式为下载

content-disposition:attachment;filename=xxx
1

‍ 为此,我们需要请求的是 Servlet 的路径,并且传参文件名,告诉 Servlet 我们要的是什么文件

<a href="/hello/downLoadServlet?fileName=1.jpg">图片1</a> <br>
<a href="/hello/downLoadServlet?fileName=1.mp4">视频1</a> <br>
1
2

‍

# 下载 Servlet

接下来就是实现 Servlet,步骤如下

  1. 获取文件名称
  2. 使用字节输入流加载文件进内存,涉及到真实路径
  3. 指定 response 的响应头: content-disposition: attachment;filename=xxx
  4. 将数据写出到 response 输出流 ‍ 完整代码:
package com.peterjxl.download;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.FileInputStream;
import java.io.IOException;

@WebServlet("/downLoadServlet")
public class DownLoadServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        this.doPost(req, resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 1. 获取请求参数
        String fileName = req.getParameter("fileName");

        // 2. 使用字节输入流加载文件进内存
        // 2.1 找到文件真实路径
        ServletContext servletContext = this.getServletContext();
        String realPath = servletContext.getRealPath("/img/" + fileName);
        // 2.2 使用字节流关联
        FileInputStream fis = new FileInputStream(realPath);

        // 3. 设置响应头
        // 3.1 设置响应头:content-type
        String mimeType = servletContext.getMimeType(fileName);
        resp.setHeader("content-type", mimeType);
        // 3.2 设置响应头打开方式:content-disposition
        resp.setHeader("content-disposition", "attachment;filename=" + fileName);
        // 4. 将输入流的数据,输出到response的输出流
        ServletOutputStream sos = resp.getOutputStream();
        byte[] buff = new byte[1024 * 8];
        int len = 0
        while ( -1 != (len = fis.read(buff))){
            sos.write(buff, 0, len);
        }
        fis.close();
    }
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44

重启 Tomcat,访问 http://localhost: 8080/hello/download.html,分别测试,可以看到有下载提示框

‍

# 如果文件名包含中文名

如果文件名带有中文,还能正常下载吗?我们来测试下。

在下载页面增加一个下载链接:

<hr>
<a href="/hello/downLoadServlet?fileName=这是什么元气美少女!.jpg">这是什么元气美少女!.jpg</a> <br>
1
2

‍ 然后准备一个图片文件到 img 目录下,并重命名为“这是什么元气美少女!.jpg”

重启 Tomcat,访问并尝试下载,很遗憾,报错了:

java.lang.IllegalArgumentException: 代码点[20,061]处的Unicode字符[这]无法编码,因为它超出了允许的0到255范围。
1

‍ 这是因为中文编码的问题。解决思路:

  1. 获取客户端使用的浏览器版本信息
  2. 根据不同的版本信息,使用 URLEncoder 来设置 filename 的编码方式 ‍ 我们首先定义一个下载工具类:
package com.peterjxl.util;

import sun.misc.BASE64Encoder;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;

public class DownLoadUtils {
    public static String getFileName(String agent, String filename) throws UnsupportedEncodingException {
        if (agent.contains("MSIE")) {
            // IE浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+", " ");
        } else if (agent.contains("Firefox")) {
            // 火狐浏览器
            BASE64Encoder base64Encoder = new BASE64Encoder();
            filename = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
        } else {
            // 其它浏览器
            filename = URLEncoder.encode(filename, "utf-8");
        }
        return filename;
    }
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

然后修改 DownloadServlet 的代码:

String agent = req.getHeader("user-agent");
fileName = DownLoadUtils.getFileName(agent, fileName);
resp.setHeader("content-disposition", "attachment;filename=" + fileName);
1
2
3

‍ 重启 Tomcat 再次访问,就可以下载了

# 总结

  1. 如果超链接指向的资源能够被浏览器解析,则会直接在浏览器中展示;如果不能解析,则弹出下载提示框
  2. 弹出下载提示框:响应头设置资源的打开方式:content-disposition: attachment; filename = xxx
  3. 注意中文问题
上次更新: 2025/5/17 12:26:09
ServletContext
Cookie 笔记

← ServletContext Cookie 笔记→

最近更新
01
吐槽一下《僵尸校园》
05-15
02
2025 年 4 月记
04-30
03
山西大同 “订婚强奸案” 将会给整个社会带来的影响有多严重? - 知乎 转载
04-26
更多文章>
Theme by Vdoing | Copyright © 2022-2025 | 粤 ICP 备 2022067627 号 -1 | 粤公网安备 44011302003646 号 | 点击查看十年之约
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式