一般浏览器内核不同,通过超链接的方式打开文件不一定会提示下载,如超链接一张图片,浏览器自动解析该图片,而不会提示下载.因此可以将超链接指向服务器Servlet,让Servlet来完成
先写一个静态HTML
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<a href="\91tangteacher\DownloadServlet?filename=黄图.jpg">高清黄图</a>
<script>
var element = document.getElementsByTagName("a")[0];
//http协议由国外定义,不支持中文,因此地址中有中文需要用某种方式转码
element.href=encodeURI(element.href);//使用js中的encodeURI方法将a标签的herf属性编码
</script>
</body>
</html>
DownloadServlet类
package com.creat.test;
import com.creat.Utils.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;
import java.io.InputStream;
import java.net.URLDecoder;
@WebServlet("/DownloadServlet")
public class DownloadServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String filename = request.getParameter("filename");//获取请求下载的文件名
String decodefilename = URLDecoder.decode(filename, "utf-8");
//java中相同编解码规则对应的类为URLencode和URLdecode
//这里使用URLDecoder类静态方法decode对文件名解码(请求路径中中文已被编码)
ServletContext servletContext = getServletContext();//获取服务器域对象
String mimeType = servletContext.getMimeType(decodefilename);//获取传输文件类型(一般文件在传输时有对应的类型,web.xml配置文件中列举了所有对应关系)
response.setHeader("Content-Type",mimeType);//设置传输文件类型
String agent = request.getHeader("User-Agent");//获取请求的浏览器类型
String encodefilename = DownLoadUtils.getFileName(agent, decodefilename);//下载弹窗中文文件名会乱码,这里使用特殊方式编码
response.setHeader("Content-Disposition","attachment;filename="+encodefilename);//设置客户端浏览器对文件以附件处理
String realPath = servletContext.getRealPath("/"+decodefilename);//域对象获取文件真实路径
FileInputStream inputStream=new FileInputStream(realPath);//创建输入流,加载要下载的文件进内存
ServletOutputStream outputStream = response.getOutputStream();//创建输出流,写文件给客户端浏览器
byte[]bys=new byte[1024*4];
int len;
while ((len=inputStream.read(bys))!=-1){
outputStream.write(bys,0,len);
}
inputStream.close();
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
点击HTML的链接,自动请求DownloadServlet完成下载
演示
点击高清黄图,自动弹出下载对话框.
下载完成
上面使用到的DownLoadUtils 工具类代码
package com.ithema.Utils;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;
import java.util.Base64;
public class DownLoadUtils {
/**
*
* @param agent User-Agent 请求头头信息,浏览器类型
* @param filename 原始中文文件名 九尾.jpg
* @return 转码之后的非中文字符串
* @throws UnsupportedEncodingException
*/
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")) {
// 火狐浏览器
Base64.Encoder encoder = Base64.getEncoder();
filename = "=?utf-8?B?" + encoder.encodeToString(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filename = URLEncoder.encode(filename, "utf-8");
}
System.out.println(filename);
return filename;
}
}