我们的需求是:提供html,图片,zip,xlsx,txt的下载功能,不管是不是浏览器能识别的内容都会以附件的形式下载到本地
一、超链接方式实现
这种的缺点就是:浏览器能解析的类型直接打开,不能解析的类型就会下载。就是当点击的只有像zip,xlsx等这种类型的文件会直接下载,其他的点击就显示了。
当点击html下载的时候,因为浏览器可以解析该类型就直接显示了
但是点击zip就会下载了。
二、使用servlet实现
那么这个就是我们在servlet里面实现了,记住代码的方式:“两个头一个流”。就是设置文件的mime类型,设置文件的下载头和设置流。
页面就直接上图了
通过servlet实现的下载,a标签中的地址是具体的servlet路径。为了获取他们的mime类型,在路径后面拼接他们具体的文件名,在servlet中通过ServletContext.getMimeType()获取mime类型。
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<!-- <meta http-equiv="refresh" content="3;url=/web02/1.html"> -->
<title>Insert title here</title>
</head>
<body>
<h2>案例二:下载功能实现</h2>
<!-- 超链接实现下载 -->
<h3>超链接实现下载</h3>
<a href="/web02/1.html">html下载</a>
<a href="/web02/1.jpg">jsp下载</a>
<a href="/web02/1.txt">txt下载</a>
<a href="/web02/1.xlsx">xlsx下载</a>
<a href="/web02/1.zip">zip下载</a>
<br>
<!-- servlet实现 -->
<h3>servlet实现</h3>
<a href="/web02/down?filename=1.html">html下载</a>
<a href="/web02/down?filename=1.jpg">jsp下载</a>
<a href="/web02/down?filename=1.txt">txt下载</a>
<a href="/web02/down?filename=1.xlsx">xlsx下载</a>
<a href="/web02/down?filename=1.zip">zip下载</a>
</body>
</html>
servlet:
package com.zmh.download;
import java.io.IOException;
import java.io.InputStream;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class DownLoad extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取文件名
String filename = request.getParameter("filename");
//获取servletContext对象
ServletContext sc = getServletContext();
//根据servletContext对象的getMimeType方法获取mime类型
String mimeType = sc.getMimeType(filename);
//设置文件的mime类型头
response.setHeader("content-type", mimeType);
//获取浏览器的版本内核
String agent = request.getHeader("user-agent");
System.out.println(agent);
// 设置文件的下载头
response.setHeader("content-disposition", "attachment;filename="+DownLoadUtils.getName(agent, filename));
// 设置流
//获取输入流
InputStream inputStream = sc.getResourceAsStream("/"+filename);
// 获取输出流
ServletOutputStream outputStream = response.getOutputStream();
//俩流互拷
byte[] buf = new byte[1024];
int len = 0;
while((len=inputStream.read(buf))!=-1) {
outputStream.write(buf, 0, len);
}
//关闭流
outputStream.close();
inputStream.close();
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
doGet(request, response);
}
}
在设置文件的下载头的时候,对当前浏览器的内核进行了判断,并且借助了DownLoadUtils这个类,对不同的浏览器的字符转换进行了设置
package com.zmh.download;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.Base64;
import java.util.Base64.Encoder;
public class DownLoadUtils {
public static String getName(String agent, String filename) throws UnsupportedEncodingException {
if (agent.contains("Firefox")) {
// 火狐浏览器
Encoder base64Encoder = Base64.getEncoder();
filename = "=?utf-8?B?" + base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filename = URLEncoder.encode(filename, "utf-8");
}
return filename;
}
}