package edu.jee.hxt.downfile;
import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.net.URLDecoder;
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;
/**
* Servlet implementation class Down11Servlet
*/
@WebServlet("/down11.do")
public class Down11Servlet extends HttpServlet {
private static final long serialVersionUID = 1L;
/**
* @see HttpServlet#HttpServlet()
*/
public Down11Servlet() {
super();
// TODO Auto-generated constructor stub
}
/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse
* response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// *****防止乱码的解决方案
// 输入数据乱码解决方案:使用request对象获取浏览器提交数据前,先设置字符集
request.setCharacterEncoding("utf-8");
// 输出数据乱码解决方案,使用request输出数据前,先设置字符集和内容类型
response.setCharacterEncoding("utf-8");
response.setContentType("text/html; charset=utf-8");
// 获取要下载的文件名称
String filename = request.getParameter("path");
System.out.println(filename);
// 中文文件名,使用url解码器解码
filename = URLDecoder.decode(filename, "UTF-8");
System.out.println(filename);
// 文件都存放在WEB-INF/uproot目录下,需要获取文件的真实路径
String filepath = request.getServletContext().getRealPath("/WEB-INF/uproot/" + filename);
// 判断filename文件是否存在,不存在,输出错误信息,
// 创建关于filename的对象
System.out.println(filepath);
File file = new File(filepath);
if (!file.exists()) {
System.out.println("您要下载的文件不存在");
response.getWriter().print("您要下载的文件不存在");
return;
}
// 所有浏览器都会使用本地编码,即中文操作系统使用GBK
// 流浪器收到这个文件名,会使用ios-8859-1来解决
filename = new String(filename.getBytes("GBK"), "ISO-8859-1");
// 添加content-disposition头,指定了在下载框中显示的文件名称
// 只要设置了这个头的信息,流浪器就会弹出下载框,就可以下载jpg和txt文件了.
response.addHeader("content-disposition", "attachment;filename=" + filename);
// 根据response.getOutputStream()获得一个ServletOutputStream类
ServletOutputStream servletOutputStream = response.getOutputStream();
// 创建一个读取文件的输入流,(节点流),把要下载的文件路径的file对象放入参数中
FileInputStream fileInputStream = new FileInputStream(file);
// 添加一个缓冲流
BufferedInputStream bufferedInputStream = new BufferedInputStream(fileInputStream);
// 用于保存实际读取的字节数
int readSize = 0;
// 创建一个竹筒,定义一个竹筒的长度
byte[] bufe = new byte[4096];
// 使用竹筒不断的从数据源取数据,一次只能取到4096长度的字节数据
// 如果没有取到数据,则返回-1
while ((readSize = bufferedInputStream.read(bufe)) != -1) {
// System.out.println("readSize");
// 使用servletOutputStream输出
// 对bufe这个字符数组,从0开始到readSize长度进行读写
servletOutputStream.write(bufe, 0, readSize);
}
// 之后记得关闭流,不然水龙头一直开着,浪费资源
servletOutputStream.flush();
servletOutputStream.close();
// 只需要对bufferedInputStream进行关闭即可
bufferedInputStream.close();
}
/**
* @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse
* response)
*/
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// TODO Auto-generated method stub
doGet(request, response);
}
}
jsp文件编辑
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>下载测试</title>
</head>
<body>
<br>当前上下文运行的物理路径:<%=application.getRealPath("/")%>
<br>注意:需要将要下载的文件放在WEB-INF/uproot/目录下。
<br>
<a href="<c:url value='/down11.do?path=黄.jpg'/>">黄.jpg</a>
</body>
</html>
在WEB-INF下必须有在Servlet所要求的文件,在这个例子中,我要求在WEB-INF下必须有uproot文件夹