原理
文件下载的实质就是文件拷贝,将文件从服务器端拷贝到浏览器端。所以文件下载需要IO技术将服务器端的文件使用InputStream读取到,在使用 ServletOutputStream写到response缓冲区中
步骤步骤
1.获得要下载的文件的名称(刚刚传过来的值)
String filename = request.getParameter("filename");
2.获得要下载的文件的类型,客户端通过文件的MIME(扩展名)去区分类型
response.setContentType(this.getServletContext().getMimeType(filename));
3.告诉客户端该文件不是直接解析,而是通过下载打开
response.setHeader("Content-Disposition", "attachment;filename="+filenameEncoder);
4.获取文件的绝对路径
String path = this.getServletContext().getRealPath("download/"+filename);
5.输入、输出流
InputStream in = new FileInputStream(path);
ServletOutputStream out = response.getOutputStream();
6.拷贝
int len = 0;
byte[] buffer = new byte[1024];
while((len=in.read(buffer))>0){
out.write(buffer, 0, len);
}
in.close();
实例:
jsp代码:
----------------------------------------------------------------------------
servet代码:
//*******文件名称是中文的下载*******
//获得要下载的文件的名称
String filename = request.getParameter("filename");//????.jpg
//解决获得中文参数的乱码----下节课讲
filename = new String(filename.getBytes("ISO8859-1"),"UTF-8");//美女.jpg(通过这种方法来获取set的值可以解决传输乱码问题)
//获得请求头中的User-Agent
String agent = request.getHeader("User-Agent");
//根据不同浏览器进行不同的编码
String filenameEncoder = "";//下面这段是针对不同浏览器的编码(编码前和编码后是两个不同的值)
if (agent.contains("MSIE")) {
// IE浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
filenameEncoder = filenameEncoder.replace("+", " ");
} else if (agent.contains("Firefox")) {
// 火狐浏览器
BASE64Encoder base64Encoder = new BASE64Encoder();
filenameEncoder = "=?utf-8?B?"
+ base64Encoder.encode(filename.getBytes("utf-8")) + "?=";
} else {
// 其它浏览器
filenameEncoder = URLEncoder.encode(filename, "utf-8");
}
//要下载的这个文件的类型-----客户端通过文件的MIME类型去区分类型
response.setContentType(this.getServletContext().getMimeType(filename));
//告诉客户端该文件不是直接解析 而是以附件形式打开(下载)----filename="+filename 客户端默认对名字进行解码
response.setHeader("Content-Disposition", "attachment;filename="+filenameEncoder);//(这里是放入浏览器当中的头文件,但是不同浏览器会有不同的接码方式,所以在这里需要进行编码(针对不同浏览器的编码))
//获取文件的绝对路径
String path = this.getServletContext().getRealPath("download/"+filename);//这里使用的是没有编码的filename,因为编码之后的filenameEncoder它因为经过不同的编码方式,相对于服务器端来说会乱码
//获得该文件的输入流
InputStream in = new FileInputStream(path);
//获得输出流---通过response获得的输出流 用于向客户端写内容
ServletOutputStream out = response.getOutputStream();
//文件拷贝的模板代码
int len = 0;
byte[] buffer = new byte[1024];
while((len=in.read(buffer))>0){
out.write(buffer, 0, len);
}
in.close();