写在前面
在实现上传和在下载之前需要做一些准备工作,去 Apache 官网下载文件上传和下载的两个组件,这里给出地址和maven 坐标.
-
common-fileupload组件下:
http://commons.apache.org/proper/commons-fileupload/
-
common-io组件下载:
http://commons.apache.org/proper/commons-io/
-
maven 坐标
<!-- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload -->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.3.1</version>
</dependency>
<!-- https://mvnrepository.com/artifact/commons-io/commons-io -->
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.2</version>
</dependency>
表单上传文件
进行文件上传时,表单需要做的准备:
-
请求方式为post
<form action="uploadServlet" method="post"……/>
-
使用file 的表单域
<input type="file" name="file"/>
3.使用multipart/form-data的请求编码方式
<form action="uploadServlet" method="post" enctype="multipart/form-data">
如下:
<form action="uploadServlet" method="post" enctype="multipart/form-data">
File:<input type="file" name="file"/>
<input type="submit" value="Submit"/>
</form>
基于表单的文件上传
- 如果在表单中使用表单元素,浏览器在解析表单时,
会自动生成一个输入框和一个按钮,输入框可供用户填写本地文件的文件名和
路径名,按钮可以让浏览器打开一个文件选择框供用户选择文件:
Enctype 属性 - 当表单需要上传文件时,需指定表单enctype 的值为multipart/form-data
- 在form 元素的语法中,enctype 属性指定将数据发送到服务器时浏览器使用的编码类型.
- enctype 属性取值
- application/x-www-form-urlencoded:表单enctype 属性的默认
值,这种编码方案使用有限的字符集,当使用了非字母和数字时,必须用
"%HH"代替(H代表十六进制数字),对于大容量的二进制数据或包含非ASCII字符的文本来说,这种编码不能满足要求 - multipart/form-data:form设定了enctype="multipart/form-data"属性后,
表示表单以为禁止传输数据.
- application/x-www-form-urlencoded:表单enctype 属性的默认
服务端
- 不能再使用request.getParameter()等方式获取请求信息,获取不到,
因为请求的编码方式已经改为multipart/form-data,以二进制的方式来提交
请求信息. - 可以使用输入流的方式来获取
- 具体使用common-fileupload 组件来完成文件的上传操作.
- 搭建环境,导入fileupload和io的包
- 基本思想:
- commons-fileupload 可以解析请求,得到一个FileItem对象组成的List
- commons-fileupload 把所有的请求信息都解析为 FileItem 对象, 无论是一个一般的文本域还是一个文件域.
- 可以调用 FileItem 的 isFormField() 方法来判断是一个表单域 或不是表单域(则是一个文件域)
- 再来进一步获取信息
实例
jsp 页面
<html>
<head>
<title>Title</title>
</head>
<body>
<form action="${pageContext.request.contextPath}/uploadServlet" method="post" enctype="multipart/form-data">
<%--文件输入--%>
File:<input type="file" name="file"/><br><br>
<%--文件描述--%>
Desc:<input type="text" name="desc"/><br><br>
<input type="submit" value="Submit"/>
</form>
</body>
</html>
UploadServlet.java
public class UploadServlet extends HttpServlet {
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//1.得到FileItem 的集合items.
DiskFileItemFactory factory = new DiskFileItemFactory();
factory.setSizeThreshold(1024*500);
//设置临时文件夹
File tempDirectory = new File("d:\\tempDirectory");
factory.setRepository(tempDirectory);
ServletFileUpload upload = new ServletFileUpload(factory);
//设置最大容量
upload.setSizeMax(1024*1024*5);
try {
List<FileItem> items = upload.parseRequest(req);
for(FileItem item:items){
//2.若是一个一般的表单域,打印信息.
if(item.isFormField()) {
String name = item.getFieldName();
String val = item.getString();
System.out.println(name+":"+val);
}
//3.若是文件域,则把文件保存到d://files目录下.
else{
String fieldName = item.getFieldName();
String name = item.getName();
String contentType = item.getContentType();
long size = item.getSize();
//获取输入流
InputStream in = item.getInputStream();
byte[] buffer = new byte[1024];
int len = 0;
name = "d:\\files\\"+ name;
//通过输出流把文件写到其它位置
FileOutputStream out = new FileOutputStream(name);
while((len = in.read(buffer)) != -1){
out.write(buffer,0,len);
}
out.close();
in.close();
}
}
} catch (FileUploadException e) {
e.printStackTrace();
}
}
文件的下载
步骤
- 设置 contentType 响应头: 设置响应的类型是什么 ? 通知浏览器是个下载的文件
response.setContentType("application/x-msdownload");
- 设置 Content-Disposition 响应头: 通知浏览器不再有浏览器来自行处理(或打开)要下载的文件, 而由用户手工完成
response.setHeader("Content-Disposition", "attachment;filename=abc.txt");
- 具体的文件: 可以调用 response.getOutputStream 的方式, 以 IO 流的方式发送给客户端.
OutputStream out = response.getOutputStream();
String pptFileName = "C:\\Users\\Think Pad\\Desktop\\__正在上课__\\11.尚硅谷_JavaWEB_监听器.pptx";
InputStream in = new FileInputStream(pptFileName);
byte [] buffer = new byte[1024];
int len = 0;
while((len = in.read(buffer)) != -1){
out.write(buffer, 0, len);
}
in.close();
实例
download.jsp
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<!-- 静态下载,xyz.txt这个文件直接放在那儿就行 -->
<a href="xyz.txt">download xyz.txt</a><br><br>
<a href="test.jsp">download test.jsp</a><br><br>
<a href="${pageContext.request.contextPath}/downloadServlet">DownLoad abcd.pptx</a>
</body>
</html>
test.jsp
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
</head>
<body>
<%
//1. 通知客户端浏览器: 这是一个需要下载的文件, 不能再按普通的 html 的方式打开.
//即设置一个响应的类型: application/x-msdownload
response.setContentType("application/x-msdownload");
//2. 通知客户端浏览器: 不再有浏览器来处理该文件, 而是交由用户自行处理
//设置用户处理的方式: 响应头: Content-Disposition
response.setHeader("Content-Disposition", "attachment;filename=abc.txt");
%>
<h4>Test Page</h4>
Time: <%= new Date() %>
</body>
</html>
downloadServlet
public class DownloadServlet extends HttpServlet {
private static final long serialVersionUID = 1L;
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.setContentType("application/x-msdownload");
//下载文件的名字
String fileName = "文件下载.pptx";
//设置响应头,控制浏览器下载该文件
response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8"));
//获取输出流
OutputStream out = response.getOutputStream();
String pptFileName = "F:\\视频\\个人博客系统_视频\\document\\基于SSM的个人博客系统.pptx";
//读取要下载的文件,保存到输入流
InputStream in = new FileInputStream(pptFileName);
//创建缓冲区
byte [] buffer = new byte[1024];
int len = 0;
//循环从输入流中的内容读取到缓冲区中!
while((len = in.read(buffer)) != -1){
out.write(buffer, 0, len);
}
in.close();
}
}