对于文件下载案例,是综合了request的相关方法、response的相关方法、IO流、消息头的获取等知识。不积跬步,无以至千里;不积小流,无以成江海。所以还是建议要踏踏实实打好基础。
先给出一个用来展示的页面
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>图片下载页面</title>
</head>
<body>
<a href="/servlet3.0/downloadServlet?filename=狗狗.jpg">下载图片</a>
</body>
</html>
核心代码来了:
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("/downloadServlet")
public class DownloadServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//1.获取请求参数,文件名称
String filename = request.getParameter("filename");
//2.使用字节输入流加载文件进内存
//2.1找到文件服务器路径
ServletContext servletContext = this.getServletContext();
String realPath = servletContext.getRealPath("/img/" + filename);
//2.2用字节流关联
FileInputStream fis = new FileInputStream(realPath);
//3.设置response的响应头
//3.1设置响应头类型:content-type
String mimeType = servletContext.getMimeType(filename);//获取文件的mime类型
response.setHeader("content-type",mimeType);
//3.2设置响应头打开方式:content-disposition
response.setHeader("content-disposition","attachment;filename="+filename);
//4.将输入流的数据写出到输出流中
ServletOutputStream sos = response.getOutputStream();
byte[] buff = new byte[1024 * 8];
int len = 0;
while((len = fis.read(buff)) != -1){
sos.write(buff,0,len);
}
fis.close();
}
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
this.doPost(request,response);
}
}
注意:
-
Content-Type的作用:该实体头的作用是让服务器告诉浏览器它发送的数据属于什么文件类型。
例如:当Content-Type 的值设置为text/html和text/plain时,前者会让浏览器把接收到的实体内容以HTML格式解析,后者会让浏览器以普通文本解析. -
Content-Disposition 的作用:当Content-Type 的类型为要下载的类型时 , 这个信息头会告诉浏览器这个文件的名字和类型
response.setHeader(“Content-Type”,“video/x-msvideo”);
response.setHeader( “Content-Disposition”, “attachment;filename=” + new String( “文件名称”.getBytes(“gb2312”), “ISO8859-1” ) ); -
8*1024 = 8K
-
byte[] b = new byte[1024] byte[] buff = new byte[1024 * 8];
事先定义了一个byte类型的数组,数组长度为1024 * 8。也就是说你最多可以存1024*8个字节的东西,如果超过这个值就会报溢出的异常了 -
while ((len = (fis.read(buff))) != -1) //不等于-1说明没有读到文件的末尾
-
将输入流中最多 len 个数据字节读入 byte 数组。尝试读取 len 个字节,但读取的字节也可能小于该值。以整数形式返回实际读取的字节数。
-
sos.write(buff,0,len); 写到 ServletOutputStream 中,每次都是从buff内存处的0偏移开始写
代码大体完成,运行起来看看
文件名称那里,显示的有点问题(应该显示中文“狗狗”,而不是" _ ")
为此,我们创建一个工具类 DownLoadUtils.java
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;
}
}
在核心代码里面加入:
//获取user-agent请求头、
String agent = request.getHeader("user-agent");
//使用工具类方法编码文件名即可
filename = DownLoadUtils.getFileName(agent, filename);
问题解决啦