1.上传(不能使用BaseServlet):
上传的作用,略
上传的要求(对表单和Servlet都有要求):
1.必须使用表单,而不能是超链接,method="post" 文件明显不能get把参数带在后面
2.必须使用多部件表单数据 enctype="multipart/form-data"
3.表单中必须添加文件表单项 input type="file" name="xxx"
4.凡是带文件的表单的所有(所有!)数据都不能用getParameter()方法获取,带了enctype
就无法使用此方法,而是使用getInputStream() 返回整个请求体
多部件表单的体:
1.一个表单项一个部件
2.一个部件中包含请求头、空行、请求体
3.普通表单项:一个头 Content-Disposition name=“xxx” 即表单项的名字,体就是表单项的值
文件表单项:包含两个头:Content-Disposition name=“xxx”,表单项名称,和一个filename="xxx"即文件名称
Content-Type:上传文件的MIME类型
commons-fileupload.jar包登场,他有一个io的依赖包
它可以帮我们解析request中的数据,一个表单数据封装到一个items对象中,调用fileItems的方法即可
上传三步:
1.工厂 DiskFileItemFactory
2.解析器 ServletFileUpload
3.表单项 FileItem
创建工厂 有构造器
创建解析器 给一个工厂,就可以new
得到表单项 使用解析器得到集合数据
给一个request List<FileItem> FileItemList = sfu.parseRequest(request);
给出一个小例子:
JSP页面:
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <%@ taglib prefix="c" uri="http://java.sun.com/jstl/core" %> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP ‘form1.jsp‘ starting page</title> <meta http-equiv="pragma" content="no-cache"> <meta http-equiv="cache-control" content="no-cache"> <meta http-equiv="expires" content="0"> <meta http-equiv="keywords" content="keyword1,keyword2,keyword3"> <meta http-equiv="description" content="This is my page"> <!-- <link rel="stylesheet" type="text/css" href="styles.css"> --> </head> <body> <form action="<c:url value=‘/UploadServlet‘/>" method="post" enctype="multipart/form-data"> 用户名:<input type="text" name="username"/><br/> 照 片:<input type="file" name="photo"/><br/> <input type="submit" value="上传"/> </form> </body> </html>
Servlet页面:
package cn.itacast.servlet; import java.io.File; import java.io.IOException; import java.util.List; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.commons.fileupload.FileItem; import org.apache.commons.fileupload.FileUploadException; import org.apache.commons.fileupload.disk.DiskFileItemFactory; import org.apache.commons.fileupload.servlet.ServletFileUpload; public class UploadServlet2 extends HttpServlet { public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { request.setCharacterEncoding("utf-8"); response.setContentType("text/html;charset=utf-8"); //上传三步,步骤见笔记 DiskFileItemFactory factory = new DiskFileItemFactory(); ServletFileUpload sfu = new ServletFileUpload(factory); sfu.setFileSizeMax(100*1024); try{ List<FileItem> FileItemList = sfu.parseRequest(request); FileItem f1 = FileItemList.get(0); FileItem f2 = FileItemList.get(1); System.out.println("普通表单项:"+f1.getFieldName()+"="+f1.getString("UTF-8")); System.out.println("文件表单项:"); System.out.println("文件类型:"+f2.getContentType()); File file = new File("F:/2.jpg"); f2.write(file); }catch(Exception e){ e.printStackTrace(); } } }
上传的细节:
1.文件必须保存到WEB-INF下(当然C.D.E盘都可以,但强烈不建议),目的是为了不让浏览器直接访问
2.文件名称相关:有的是绝对路径,需要进行切割(切割文件名过来),
部分浏览器(IE6,遨游之类)这样
Sting filename = f2.getName();
int index = filename.lastIndexOf("\\");
if(index != -1){
String filename1 = filename.subString(index+1);
}
3.乱码问题:文件的名称乱码等都交给FileUpload来处理,使用时告诉它编码,request.setCharaterEncoding("UTF-8");
fileUpload会自动调用处理
当然,解析器也提供了一个方法
ServletFileUpload.setHeaderEncoding();内部的方法,优先级高
4.处理重名问题,为每个文件名称添加前缀(UUID),处理重名问题
filename = CommonUtils.uuid()+"_"+filename;使用客户的文件名称再加我们的不重复前缀
5.目录打散问题:不能在一个文件夹下存放过多文件
首字母打散:如文件名称叫abc.txt 则保存到B目录下,不存在则创建
局限也明显,中文等的太多复杂
时间打散:使用当前日期作为目录,一天一个文件夹
局限性:有时候周末目录爆满,工作日空空如也
哈希打散(推荐):任何对象都有一个hashCode()方法,得到一个INT值
得到哈希值
把Int值转换成十六进制
获取十六进制的前两位用来生成目录
目录为两层,例如文件名 1A2B3C 生成目录 1/A/文件名
最多可以生成INT的范围,4字节32位,8位十六进制
最多42亿
局限性体现在人工不知道在哪,找起来麻烦,但计算机很清楚
切割文件名->加UUID->得到文件名的哈希值->Integer.toHexString()
->使用String.cahrAt()拿出构建目录的目录名+root文件路径("/WEB-INF/files/") mkdirs
6.上传文件大小限制:单个文件大小限制
整个请求大小限制
解析器有相关的方法:sfu.setFilesizeMax(100*1024);100KB
必须放在解析之前
如果超出限制,在解析时会抛出异常
限制整个表单请求大小 sfu.setSizeMax();也是解析前执行
7.缓存大小与临时目录
1.缓存大小:超出多大才向硬盘保存,默认10KB
2.向硬盘的什么目录保存
DiskFileItemFactory(int sizeThreshold, File repository)
使用此构造器,可以指定缓存与临时目录
2.下载:下载就是客户端响应字节数据
下载的要求:
两个头一个流
Content-Type (什么类型)
Content-Dispositon (默认值是inline,浏览器能打开他就打开,图片,文字等)
要弹下载框需要设置为 accathment;filename=xxx
流就是要下载的文件数据
下载的细节:
显示在下载框的中文名称会出现乱码
大部分主流浏览器都使用URL编码
通用解决方案:frameName = new String(filename.getBytes(""),"");
一个小工具类(此处略) filenameEncoding方法