上传不能使用BaseServlet 因为无法获取参数了getParameter 方法不能用
文件上传概述
1、文件上传的作用
例如网络硬盘,就是用来上传下载文件的
2、文件上传也对面的要求
1、必须使用表单,而不能是超链接
2、表单的method必须是POST,而不能是GET
3、表单的enctype必须是multipart/form-data; 处理编码
4、在表单中添加file表单字段,即<input type="file"name="xxx"/>
<form action="xxx" method ="post" enctype="multipart/form-data">
<input type="text" name="username"/>
<input type="text" name="username"/>
3、上次对Servlet限制
request.getParameter("xxx");这个方法在表单为enctype="multipart/form-data"时,作废了,永远返回null
ServletInputStream request.getInputStream
();包含整个请求的体
多部件表单的体
1、每隔出多个部件,即一个表单项一个部件
2、一个部件中自己包含请求头和空行,以及请求体
3、普通表单项
1个头:Content-Disposition 包含name="xx",即表单项名称
体就是表单项的值
4、文件表单项
2个头
Content-Disposition:包含name="xxx"即表单项名称:还有一个filename="xxx",表示上次文件的名称
Content-Type:是上传文件的MIME类型 例如:image/pjpeg,表示上传的是图片,图上中jpg扩展名的图片
体就是上传文件的内容
commons-fileupload
commons-fileupload.jar 上传依赖io
commons-io.jar 可以独立存
可以帮我们解析request中的上传数据,解析后的结果是一个表单项封装到一个FileItem对象中,只需要调用FileItem的方法即可
1、上传3步
相关类:
工厂:DiskFileItemFactory
解析器:ServletFileUpload
表单项:FileItem
1、创建工厂:DiskFileItemFactory factory = new 工厂:DiskFileItemFactory();
2、创建解析器:ServletFileUpload sfu = new 解析器:ServletFileUpload(factory);
3、使用解析器来解析request:得到FileItem集合:List<FileItem> fileItemList = sfu.paraseRequest(request);
2、FileItem
boolean isFormField():是否为普通表单项。返回true为普通表单项,如果为false即文件表单项
String getFieldName():返回当前表单项的名称
String getString(String charset):返回表单项的值
String getName():返回上传的文件名称
Long getSize():返回上传文件的字节数
InputStream getInputStream():返回上传文件对应的输入流
void write(File destFile):把上传的文件内容保存到指定的文件中
String getContentType():获取文件类型
上传的细节
1、文件必须保存到WEB-INF下
不让浏览器直接访问到
假如说用户上传了一个a.jsp文件,然后用户在通过浏览器去房屋呢这个a.jsp文件,那么就会执行a.jsp中的内容。
如果在a.jsp中有Runtime.getRuntime().exec("shutdown -s -t 1");
shutdown -a 取消
通常会在WEB-INF目录下创建一个uploads目录来存放上传的文件,而在servlet中找到这个目录需要使用ServletContext的getRealPath(Sting)方法,
ServletContext servletContext = this.getServletContext();
String savapath = servletContext.getRealPath("WEB-INF/uploads");
其中savapath为:F:\tomcat8_5\webapps\uploads\WEB-INF\uploads
2、文件名称相关问题
有的浏览器(IE6)上传的文件名是绝对路径,需要切割 c:\files\aa.jpg
无论是否为完整路径,都去截取最后一个"\\"后面的内容即可
String filename = fi2.getName();
int index = filename.lastIndexOf("\\");
if(index!=-1){
filename = filename.subString(index+1);
}
文件名乱码或者普通表单项乱码。request.setCharacterEncoding("utf-8");因为fileUpload内部会调用request.getCharacterEncoding("utf-8")
ServletFileUpload.setHeaderEncoding("utf-8");//优先级高
文件同名问题:需要为每个文件添加名称前缀:这个前缀要保证不能重复。uuid
filename = CommonUtils.uuid() + "_" + filename;
3、目录打散
不能在一个目录下存放过多文件 过万文件在一个文件夹会很卡
首字符打散:使用文件的首字母作为目录名称,例如abc.txt.那么我们把文件保存到a目录下,如果a目录此时不存在,那么创建
时间打散:使用当前日期作为目录
哈希打散:
通过文件名称得到int值,即调用hashCode()
它int值转换成16进制0~9,A~F
获取16进制的前两位来生成目录,目录为二层。例如1B2C3D4E5F /1/B保存文件
4、上传文件的大小限制
单个文件大小限制 ServletFileUpload类的setFileSizeMax(long)即可,参数就是上传文件的上限字节数,
例如servletFileUpload.setFileSizeMax(1024*10)表示上限为10kb
一旦超出会抛出FileUploadBase.FileSizeLimitExceededException
可以在sevlet获取这个异常然后向页面输出
必须在解析开始之前 parseRequest之前
如果上传的文件超出限制,在parseRequest()方法执行时,会抛出异常FileUploadBase.FileSizeLimitExceededException
整个请求所有数据大小限制
sfu.setSizeMax(1024*1024);//限制整个表单大小为1M
这个方法也是必须在parseRequest()之前调用
如果上传的文件超出限制,在parseRequest()方法执行时,会抛出异常FileUploadBase.SizeLimitExceededException
5、缓存大小与临时目录
缓存大小:超出多大,才向硬盘保存。默认10KB
临时目录:向硬盘的什么位置保存
设置缓存大小与临时目录:new DiskFileItemFactory(20*1024,new Fil("F:/temp"))
form.jsp
<h1>上传</h1>
<form action="<c:url value='/UploadServlet'/>" method="post" enctype="multipart/form-data">
用户名:<input type="text" name="username"/><br/>
照片<input type="file" name="picture"><br/>
<input type="submit" value="上传"/><%--文件表单项 --%>
</form>
public class UploadServlet extends HttpServlet {
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
/*
* 上传三步
* 1、得到工厂
* 2、通过工厂创建解析器
* 3、解析request,得到FileItem集合
* 4、遍历FileItem集合,调用其API完成文件的保存
*/
DiskFileItemFactory factory = new DiskFileItemFactory();
ServletFileUpload sfu = new ServletFileUpload(factory);
try {
List<FileItem> fileItemList = sfu.parseRequest(request);
FileItem fi1 = fileItemList.get(0);
FileItem fi2 = fileItemList.get(1);
System.out.println("普通表单项"+fi1.getFieldName()+"="+fi1.getString("utf-8"));
System.out.println("文件表单项"+"Contetn-type"+fi2.getContentType()+"size"+
fi1.getSize()+"filename"+fi2.getName());
//保存文件
File destFile = new File("c:/aa.jpg");
fi2.write(destFile);
} catch (FileUploadException e) {
throw new RuntimeException(e);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
}
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");
ServletInputStream in = request.getInputStream();
String s = IOUtils.toString(in);
System.out.println(s);
}
}
public class UploadServlet3 extends HttpServlet {
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
/*
* 1、创建工厂
* 2、创建解析器
* 3、解析器解析
*/
//工厂
DiskFileItemFactory factory = new DiskFileItemFactory(20*1024,new File("G:/temp"));
//解析器
ServletFileUpload sfu = new ServletFileUpload(factory);
// sfu.setFileSizeMax(100*1024);//限制单个文件大小为100KB
// sfu.setSizeMax(1024*1024);//限制整个表单大小为1M
//解析得到List
try {
List<FileItem> list = sfu.parseRequest(request);
FileItem fi = list.get(1);
/*
* 1、得到文件保存的路径
*/
String root = this.getServletContext().getRealPath("/WEB-INF/files/");
/*
* 2、生成两层目录
* 1、得到文件名称
* 2、得到hashCode
* 3、转换成16进制
* 4、获取前2个字符用来生成目录
*/
String filename = fi.getName();//获取上传的文件名称
/*
* 处理文件名的绝对路径问题
*/
int index = filename.lastIndexOf("\\");
if(index != -1){
filename = filename.substring(index+1);
}
/*
* 给文件名称添加uuid前缀,处理文件同名问题
*/
String savename = CommonUtils.uuid()+"_"+filename;
/*
* 1、得到hashCode
*/
int hCode = filename.hashCode();
String hex = Integer.toHexString(hCode);
/*
* 2、获取hex的前两个字母,与root连接在一起,生成一个完整的路径
*/
File dirFile = new File(root,"/"+hex.charAt(0)+"/"+hex.charAt(1));
/*
* 3、创建目录链
*/
dirFile.mkdirs();
/*
* 4、创建目录文件
*/
File destFile = new File(dirFile,savename);
/*
* 5、保存
*/
fi.write(destFile);
} catch (FileUploadException e) {
if(e instanceof FileUploadBase.FileSizeLimitExceededException){
request.setAttribute("msg", "上传的文件超过了100kb");
request.getRequestDispatcher("/form3.jsp").forward(request, response);
}
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}