1.文件上传(浏览器-->服务器)
* 上传功能
* 浏览器可以选择文件
* 浏览器可以将选择的文件上传给服务器
* 服务器需要将上传文件保存到服务器本地
* 上传基本要求
* 1.提供input标签type属性的值,必须为file(不同的浏览器实现的效果不同)
* 注意:input必须设置name属性,否则浏览器将不会发送内容
* 2.可以上传
* input必须放置在表单中
* 必须将表单的method设置成post(get提交的数据有限,不能提交数据,只能提交文件的路径)
* 必须将表单的enctype设置为multipart/form-data
* 3.服务器获得数据
* request.getInputStream()
* 表单form的enctype属性的取值
* application/x-www-form-urlencoded 在发送前编码所有字符(默认)
* multipart/form-data 不对字符编码。在使用包含文件上传控件的表单时,必须使用该值。
* text/plain 空格转换为 "+" 加号,但不对特殊字符编码。
2.手动上传必须繁琐,使用第三方jar包
* Servlet3.0(java ee 6.0)支持文件上传
* apache基金组织commons--fileupload
* 核心jar包:commons-fileupload-1.2.2.jar , 来自于commons-fileupload-1.2.2-bin.zip
* 依赖jar包:commons-io-2.3.jar ,来自于commons-io-2.3-bin.zip
* 上传核心文件:ServletFileUpload
* 编写流程
* 1.提供文件列表工厂FileItemFactory
* FileItemFactory factory = new DiskFileItemFactory();
* 2.核心文件
* ServletFileUpload fileUpload = new ServletFileUpload(factory);
* 3.获得所有内容
* List<FileItem> fileItemList = fileUpload.parseRequest(request);
* 4.判断是否是普通字段
if(fileItem.isFormField()){
* 获得字段名称:fileItem.getFieldName()
* 获得字段内容:fileItem.getString();
} else {
* 获得上传文件名:fileItem.getName()
* 获得单个上传文件流:fileItem.getInputStream()
}
3.API详解
* FileItemFactory
* 构造
* 默认构造:DiskFileItemFactory()
* 有参数构造:DiskFileItemFactory(int sizeThreshold, File repository)
* 参数1:sizeThreshold,设置产生临时文件的临界大小值。如果上传的文件大于了指定的大小,将生成临时文件。
* 参数2:repository,用于指定临时文件存在的位置。默认:%tomcat%/temp
* API
* void setRepository(File repository) 位置
* void setSizeThreshold(int sizeThreshold) ,设置产生临时文件的临界大小值
* ServletFileUpload
* 用来判断是否支持文件上传:static boolean isMultipartContent(javax.servlet.http.HttpServletRequest request)
* 判断请求头中是否存在,multipart/form-data
* 获得所有的上传内容:List parseRequest(javax.servlet.http.HttpServletRequest request)
* 设置上传文件的编码:setHeaderEncoding(String encoding)
* request的编码将其作用:request.setCharacterEncoding("");\
* 设置当个上传文件的大小: void setFileSizeMax(long fileSizeMax)
* 设置总共上传文件的大小:void setSizeMax(long sizeMax)
* 设置上传文件的进度监听器:setProgressListener(ProgressListener pListener)
* FileItem
* 判断是否是普通字段:isFormField()
* 上传文件的大小:getSize()
* 普通字段的名称:getFieldName()
* 普通字段的内容:getString()
* getString(String encoding) ,在获得数据时,需要进行编码
* 上传文件的名称:getName()
* 上传文件的内容:getInputStream()
* 删除临时文件:delete();
4. 上传问题
* 1上传文件名的中文乱码:ServletFileUpload.setHeaderEncoding(String encoding)
* 2普通字段的中文乱码:FileItem.getString(String encoding)
* 3不同浏览器的上传文件名:fileName = fileName.substring(fileName.lastIndexOf(File.separator) + 1);
* 4文件名重复:fileName = UUID + "_" + fileName;
5. 上传进度的监听器
6.上传文件的位置
* 公共的图片等资源,将放置在web项目下,但出WEB-INF、META-INF目录
* 如果需要验证的资源,可以考虑WEB-INF、META-INF,也可以放置到服务器的其他位置
* 单个文件夹中文件的个数,采用目录分级管理
int hashCode = fileName.hashCode();
//第一层目录
int dir1 = hashCode & 0xf;
//第二层目录
int dir2 = (hashCode >>> 4) & 0xf;
System.out.println(dir1 + File.separator + dir2);
public class MyUploadServlet2 extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doPost(request, response);
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
try {
//防止恶意破坏
if(!ServletFileUpload.isMultipartContent(request)){
//不支持
throw new RuntimeException("必须支持文件上传");
}
//1、设置文件工厂
DiskFileItemFactory fileItemFactory = new DiskFileItemFactory();
/**
* setSizeThreshold(int sizeThreshold):设置产生临时文件的临界值大小,
* 如果上传的文件大于临界值得大小,name就产生临时文件,此方法不是有继承于父类
*/
fileItemFactory.setSizeThreshold(1024 * 1024 * 3); //3Mb
//指定临时文件的位置
fileItemFactory.setRepository(new File(this.getServletContext().getRealPath("/temp")));
//2、核心文
ServletFileUpload fileUpload = new ServletFileUpload(fileItemFactory);
//设置上传文件的编码:
fileUpload.setHeaderEncoding("utf-8");
//设置的单个文件上传德尔大小
//fileUpload.setSizeMax(1024 * 1024 * 6 );//上传最大文件的值为6Mb
MyProgressListener progressListener = new MyProgressListener();
request.getSession().setAttribute("progressListener", progressListener);
// * 注册监听
fileUpload.setProgressListener(progressListener);
//解析reqest,获取上传文件的内容
List<FileItem> fileItemList = fileUpload.parseRequest(request);
//遍历结果
for(int i = 0 ; i < fileItemList.size(); i++){
//获取每一个文件列表
/*
* FileItem:是一个接口类,这个类代表一个文件或者一个表单列表,
* 接受一个multipart/form-data POST的请求
*/
FileItem fileItem = fileItemList.get(i);
//判断是否是表单字段
//isFormFiled
if(fileItem.isFormField()){
//这里判断的结果是表单字段
String fileName = fileItem.getFieldName();//获取字段名称
System.out.println("普通字段名称:--->" + fileName);
//获得字段的内容
//String value = fileItem.getString();
String value = fileItem.getString("utf-8");
System.out.println("获取普通字段的内容:--->" + value);
}else{//这里不是表单字段
//文件上传字段
//获取上传文件字段
String fileName = fileItem.getName();
System.out.println("@@@没有处理之前的上传文件的名称:---->>" + fileName);
//##处理不同浏览器上传的文成名称
/**
* IE8之间,获取的是文件路径
* 其他浏览器,获取的是文件名称
*
* 如果File.sparator没有找到分隔符时,返回的是-1
* f:\abc.txt asc.txt
*/
fileName = fileName.substring(fileName.lastIndexOf(File.separator) + 1);
System.out.println("兼容浏览器的上传文件的名称:---->>" + fileName);
//##文件重名问题
fileName = UUID.randomUUID().toString().replace("-", "") + "_" + fileName;
//获取上传文件的内容[使用到上传文件流]
InputStream is = fileItem.getInputStream();
System.out.println("上传文件的内容:---->>");
//##上传文件的位置
File fileDir = new File(this.getServletContext().getRealPath("/upload"));
//拼接文件名
File file = new File(fileDir,fileName);
//获取文件输出流
OutputStream fileOut = new FileOutputStream(file);
//读取文件
byte[] buf = new byte[1024];
int len = -1 ;
while((len = is.read(buf)) != -1){
fileOut.write(buf,0,len);
}
fileOut.close();
is.close();
//上传成功之后,删除临时文件
fileItem.delete();
}
}
} catch (FileUploadException e) {
throw new RuntimeException(e.getMessage(),e);
}
}
}