在我的上篇博客中写了文件的上传,由于篇幅较长,分成两部分来写,内容很多是借鉴大神的,如有错误,请各位指点。
接下来是文件的下载
文件下载大致步骤如下:
1、既然下载文件,必定要得到所有的文件下载列表,即去WEB-INF文件下去递归遍历出所有的文件,将它显示在页面上
具体实现:
1、首先得到文件的WEB-INF目录,我们暂且叫它根目录;
2、到目录下去查找文件,由于目录下有很多文件夹目录,所以使用递归遍历文件夹,得到文件;
3、得到文件后,取得它的名字,但是考虑到我们上传文件时为了防止文件名重复,使用hash编码给文件名进行了扩展,所以此时我们需要将文件的真实文件名抽取出来,但是我们也要保留之前的文件名,以便我们通过他的文件名在服务器中去寻找它,为此,我们是用来Map<String,String>来存储文件在服务器中的文件名作为key,将真实文件名作为value用来显示到页面,通过下面的jsp页面可以看出其中的逻辑,然后将存储好的Map对象发送给页面,页面循环输出显示
2、页面显示出文件列表了,接下来就下载吧
具体实现:
1、通过地址栏附带需要下载的文件名(注意这里传过去的是key)到servlet中
2、servlet通过文件名和根目录,找到文件在服务器中的真实路径(因为上传文件时根据文件名和根目录利用hash算法对文件进行了层层包装,所以此时,再次根据文件名和根目录利用hash算法还原出真实路径,从而达到查找文件路径的目的)
3、查找到了文件路径(此时的路径未加文件名)之后,那么根据文件名得到文件File
4、将文件输入流嫁接到File文件对象上得到输入流,输入到内存buff
5、利用文件输出流将buff内的内容输出到页面来下载
jsp页面:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8" isELIgnored="false"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>down</title>
<meta http-equiv="Content-Type" content="text/html charset=UTF-8">
</head>
<body>
<!-- 遍历Map集合 -->
<c:forEach var="me" items="${fileNameMap}">
<c:url value="/FileDownhandServlet" var="downurl">
<c:param name="filename" value="${me.key}"></c:param>
</c:url>
${me.value}<a href="${downurl}">down</a>
<br/>
</c:forEach>
</body>
</html>
ListFileServlet:用于将WEB-INF下面的文件全部显示到jsp页面
package FileUpAndDown;
import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* Servlet implementation class ListFileServlet
*/
@WebServlet("/ListFileServlet")
public class ListFileServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
/**
* @see HttpServlet#HttpServlet()
*/
public ListFileServlet() {
super();
// TODO Auto-generated constructor stub
}
/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
response.getWriter().append("Served at: ").append(request.getContextPath());
//获取上传文件目录
String uploadFilePath = this.getServletContext().getRealPath("/WEB-INF/upload");
//存储要下载的文件名
Map<String,String> fileNameMap = new HashMap<String,String>();
//递归遍历filepath目录下的所有文件和目录,将文件的文件名存储到map集合中
listfile(new File(uploadFilePath),fileNameMap);//File既可以代表一个文件也可以代表一个目录
//将Map集合发送到listfile.jsp页面进行显示
request.setAttribute("fileNameMap", fileNameMap);
request.getRequestDispatcher("/jsp/listfile.jsp").forward(request, response);
}
private void listfile(File file, Map<String, String> fileNameMap) {
// TODO Auto-generated method stub
//如果file代表的不是一个文件,而是一个目录
if(!file.isFile()){
//列出该目录下的所有文件和目录
File files[] = file.listFiles();
//遍历files[]数组
for(File f : files){
//递归
listfile(f,fileNameMap);
}
}else{
/**
* 处理文件名,上传后的文件是以uuid_文件名的形式去重新命名的,去除文件名的uuid_部分
file.getName().indexOf("_")检索字符串中第一次出现"_"字符的位置,如果文件名类似于:9349249849-88343-8344_阿_凡_达.avi
那么file.getName().substring(file.getName().indexOf("_")+1)处理之后就可以得到阿_凡_达.avi部分
String对象的substring()方法(字符串截取)有两种:
1、substring(x):从x处开始截取字符串
2、substring(x,y):从x处开始截取,到y处停止
*/
String realName = file.getName().substring(file.getName().indexOf("_")+1);
//file.getName()得到的是文件的原始名称,这个名称是唯一的,因此可以作为key,realName是处理过后的名称,有可能会重复
fileNameMap.put(file.getName(), realName);
}
}
/**
* @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);
}
}
FileDownhanderServlet:用于下载资源
package FileUpAndDown;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.URLEncoder;
import java.nio.Buffer;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.jasper.tagplugins.jstl.core.Out;
import com.sun.jndi.toolkit.ctx.StringHeadTail;
/**
* Servlet implementation class FileDownhandServlet
*/
@WebServlet("/FileDownhandServlet")
public class FileDownhanderServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
/**
* @see HttpServlet#HttpServlet()
*/
public FileDownhanderServlet() {
super();
// TODO Auto-generated constructor stub
}
/**
* @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
*/
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
//response.getWriter().append("Served at: ").append(request.getContextPath());
//得到要下载的文件名,即jsp页面的me.key,存储的是文件的完整名
String fileName = request.getParameter("filename");
//设置字符编码
fileName = new String(fileName.getBytes("iso8859-1"),"UTF-8");
//上传的文件都是保存在/WEB-INF/upload目录下的子目录当中
String fileSaveRootPath=this.getServletContext().getRealPath("/WEB-INF/upload");
//通过文件名和文件根目录找出该文件所在目录
String path = findFileSavePathByFileName(fileName,fileSaveRootPath);
//得到要下载的文件
File file = new File(path + "\\" + fileName);
//如果文件不存在
if(!file.exists()){
request.setAttribute("message", "您要下载的资源已被删除!!");
request.getRequestDispatcher("/jsp/message.jsp").forward(request, response);
return;
}
//处理文件名
String realname = fileName.substring(fileName.indexOf("_")+1);
//设置响应头,控制浏览器下载该文件
response.setHeader("content-disposition", "attachment;filename=" + URLEncoder.encode(realname, "UTF-8"));
//得到文件输入流
FileInputStream in = new FileInputStream(file);
//缓冲区
byte buff[] = new byte[1024];
//得到文件输出流
OutputStream out = response.getOutputStream();
int len=0;
while ((len=in.read(buff))>0) {
//将文件输出到页面
out.write(buff, 0, len);
}
in.close();
out.close();
}
//通过文件名和文件保存根目录,获得上传后文件的保存路径,此过程与文件上传相同
private String findFileSavePathByFileName(String fileName, String fileSaveRootPath) {
// TODO Auto-generated method stub
//得到文件名的hashCode的值,得到的就是filename这个字符串对象在内存中的地址
int hashcode = fileName.hashCode();
int dir1 = hashcode&0xf; //0--15
int dir2 = (hashcode&0xf0)>>4; //0-15
//构造新的保存目录
String dir = fileSaveRootPath + "\\" + dir1 + "\\" + dir2; //upload\2\3 upload\3\5
//File既可以代表文件也可以代表目录
File file = new File(dir);
//如果目录不存在
if(!file.exists()){
//创建目录
file.mkdirs();
}
return dir;
}
/**
* @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
*/
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
response.setContentType("text/html;charset=utf-8");//设置响应到页面的文本编码
request.setCharacterEncoding("utf-8");//设置客户端传过来的字符编码
doGet(request, response);
}
}
web.xml:
<servlet>
<servlet-name>FileDownhanderServlet</servlet-name>
<servlet-class>javaWebTest.FileUpAndDown.FileDownhanderServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>FileDownhanderServlet</servlet-name>
<url-pattern>/FileDownhanderServlet</url-pattern>
</servlet-mapping>