HttpServletResponse-完成文件下载

本文详细介绍了文件下载的技术实现过程,包括如何使用IO技术从服务器端读取文件并发送到客户端,以及如何处理不同浏览器下的文件名编码问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

原理


文件下载的实质就是文件拷贝,将文件从服务器端拷贝到浏览器端。所以文件下载需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代码:

<a href="/WEB14/downloadServlet?filename=a.flv">a.flv</a><br><!-- 会访问DownloadServlet这个类,并传一个值 -->

<a href="/WEB14/downloadServlet?filename=a.jpg">a.jpg</a><br>

<a href="/WEB14/downloadServlet?filename=a.mp3">a.mp3</a><br>

----------------------------------------------------------------------------

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();


在Java编程中,我们通常会涉及到处理文件下载的情况。而在使用Servlet API时,我们可以使用HttpServletResponse来向客户端发送文件下载请求。在这种情况下,我们通常需要实现以下步骤: 1. 设置响应内容类型(MIME类型):我们需要设置content-type属性,以便浏览器知道该下载内容的格式。例如,如果要下载的是PDF文件,则content-type属性应设置为“application/pdf”。 2. 设置响应头(Header)信息:通过设置Content-Disposition响应头信息,我们可以确保文件下载时会自动以指定的名称进行保存,而不是提供默认的名称。 3. 生成输入流:使用Java IO流读取所需的文件,并将其写入HttpServletResponse的输出流中。 4. 完成响应:在文件输出流完成后,我们需要关闭输出流并调用response.flushBuffer() 方法,以确保数据完成从客户端的缓冲区到达浏览器之前被刷新。 具体的代码实现可以参考以下例子: ``` protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //获取要下载文件路径 String filePath = "/path/to/file.pdf"; File downloadFile = new File(filePath); FileInputStream inStream = new FileInputStream(downloadFile); //设置响应类型和头信息 response.setContentType("application/pdf"); response.setContentLength((int) downloadFile.length()); response.setHeader("Content-Disposition", "attachment; filename=" + downloadFile.getName()); //将文件内容写入输出流 byte[] buffer = new byte[4096]; int bytesRead = -1; OutputStream outStream = response.getOutputStream(); while ((bytesRead = inStream.read(buffer)) != -1) { outStream.write(buffer, 0, bytesRead); } inStream.close(); outStream.flush(); outStream.close(); } ``` 在这个例子中,我们读取名为“file.pdf”的文件,将其内容写入响应的输出流中,同时设置content-type属性和Content-Disposition响应头信息。这将确保浏览器会自动下载具有指定名称的文件。最后,我们需要关闭输入和输出流,并确保数据已刷新到客户端。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值