- 需求
- 分析
- 实现步骤
- 示例代码
- 下载的文件中文名显示问题
1.页面显示超链接 2.点击超链接后弹出下载对话框 3.完成图片文件下载
分析超链接指向的资源如果能够被浏览器解析,则直接在浏览器中展示,如果不能解析,则弹出下载提示框。如果希望无论返回什么资源都不要解析,而是让用户下载,那么就必须使用 content-disposition 响应头告诉客户端,返回的资源(响应体的数据)以附件的形式打开。
实现步骤1.开发前端页面,使用超链接标签,href 属性指向一个 Servlet,并且传递下载资源的名称 2.实现后端逻辑,定义一个 Servlet,获取下载资源的名称,再根据资源名称将对应的资源加载入内存中,再从内存输出到 Response对象中,再返回给客户端 2.1.获取参数 2.2.使用字节输入流读取资源文件,加载进内存中 2.3.设置响应头(content-type 和 content-disposition) 2.4.使用字节输出流写入到 Response 对象中。具体是从内存写入到 Response 的字节输出流中,再从字节输出流写入到 Response 对象中 2.5.服务器从 Response 对象中获取数据构建成响应报文发送给客户端(这步服务器自动完成)
说明:
response.setHeader("content-disposition", "attachement;filename=123.jpeg");
上述代码表示将响应头 content-disposition 的值设为 attachement;filename=123.jpeg。其中 attachement 表示响应的资源以附件形式打开;filename 是弹出的下载对话框中显示的文件名称以及文件下载后的文件名称。
示例代码前端页面示例代码:
Title
图片
视频
服务端示例代码:
package priv.lwx.javaex.servlet_demo.web.servlet.download;
/**
* description
*
* @author liaowenxiong
* @date 2022/1/12 21:57
*/
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
@WebServlet("/download")
public class DownloadServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.获取请求参数fileName
String fileName = request.getParameter("fileName");
// 2.使用字节输入流加载文件进内存
// 2.1.获取文件的服务器路径
String realPath = this.getServletContext().getRealPath("/image/" + fileName);
// 2.2.获取文件的文件字节输入流(用文件字节流关联文件)
FileInputStream fis = new FileInputStream(realPath); // 这步就完成了读取文件数据进内存的操作
System.out.println(fis);
// 3.设置响应头
// 3.1.设置响应头content-type
ServletContext context = this.getServletContext();
// 获取文件的MIME类型
String mimeType = context.getMimeType(fileName);
// 告诉客户端响应体的数据类型
response.setHeader("content-type", mimeType);
// 3.2.设置响应头content-disposition
response.setHeader("content-disposition", "attachement;filename=" + fileName);
// 4.将输入流的数据写入到输出流中(类似文件复制的操作)
// 4.1.获取字节输出流
ServletOutputStream sos = response.getOutputStream();
// 4.2.定义读取到的字节数据的缓冲区
byte[] buff = new byte[1024 * 8];
int len = 0;
// 将字节数据读取到字节数组中,返回读取到的字节数,如果返回的字节数不是-1,说明没有读到文件末尾
// 注意:字节输入流每次读取到字节数组中的数据会覆盖原来的旧数据
while ((len = fis.read(buff)) != -1) {
// 将字节数组中的字节数据写入到字节输出流中,
sos.write(buff, 0, len);
}
// 5.释放资源
fis.close();
}
}
下载的文件中文名显示问题
如果你下载的文件名称是中文,不同浏览器版本会有不同的显示,不过都是错误的显示。
解决的办法:判断浏览器的版本,根据不同的浏览器版本对文件名进行不同的编码,再返回给客户端显示。
处理文件名称的示例代码如下:
package priv.lwx.javaex.servlet_demo.util;
import Decoder.BASE64Encoder;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
/**
* 处理下载文件的中文名无法显示的问题
*
* @author liaowenxiong
* @date 2022/1/13 11:55
*/
public class DownloadUtils {
public static String getFileName(String agent, String filename) throws UnsupportedEncodingException, UnsupportedEncodingException {
if (agent.contains("МЅIЕ")) {
// 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;
}
}
最终完整的文件下载的示例代码:
package priv.lwx.javaex.servlet_demo.web.servlet.download;
/**
* 文件下载的示例代码
*
* @author liaowenxiong
* @date 2022/1/12 21:57
*/
import priv.lwx.javaex.servlet_demo.util.DownloadUtils;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
@WebServlet("/download")
public class DownloadServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 1.获取请求参数fileName
String fileName = request.getParameter("fileName");
// 2.使用字节输入流加载文件进内存
// 2.1.获取文件的服务器路径
String realPath = this.getServletContext().getRealPath("/image/" + fileName);
// 2.2.获取文件的文件字节输入流(用文件字节流关联文件)
FileInputStream fis = new FileInputStream(realPath); // 这步就完成了读取文件数据进内存的操作
// System.out.println(fis);
// 3.设置响应头
// 3.1.设置响应头content-type
ServletContext context = this.getServletContext();
// 获取文件的MIME类型
String mimeType = context.getMimeType(fileName);
// 告诉客户端响应体的数据类型
response.setHeader("content-type", mimeType);
// 3.2.设置响应头content-disposition
// 获取请求头user-agent
/* String agent = request.getHeader("user-agent");
// 根据浏览器版本对文件名进行不同的处理,以解决中文无法显示的问题
fileName = DownloadUtils.getFileName(agent, fileName);*/
response.setHeader("content-disposition", "attachement;filename=" + fileName);// filename是在下载提示框显示的文件名
// 4.将输入流的数据写入到输出流中(类似文件复制的操作)
// 4.1.获取字节输出流
ServletOutputStream sos = response.getOutputStream();
// 4.2.定义读取到的字节数据的缓冲区
byte[] buff = new byte[1024 * 8];
int len = 0;
// 将字节数据读取到字节数组中,返回读取到的字节数,如果返回的字节数不是-1,说明没有读到文件末尾
// 注意:字节输入流每次读取到字节数组中的数据会覆盖原来的旧数据
while ((len = fis.read(buff)) != -1) {
// 将字节数组中的字节数据写入到字节输出流中,
sos.write(buff, 0, len);
}
// 5.释放资源
fis.close();
}
}