1.上传对表单的要求:
* method=“post”
* enctype=“multipart/form-data”
>每个表单项都自己是一个独立的请求格式:
*有自己的请求头:Content-Disposition,包含表单项的名称
* 空行
* 请求体
>如果表单项是文件表单项
* 有两个请求头
> Content-Disposition:
* 表单项的名称
*上传文件的名称
> Content-Type:上传文件的mime类型
* 请求体内容为文件内容!
* <input type="file" />
=================================================
上传对servlet的要求:
当使用了enctype=“multipart/form-data”
>String getParameter():所有获取请求参数的方法不能再使用!
*BaseServlet: 它依赖request.getParameter(),所以上传时也不能使用BaseServlet
> ServletInputStream getInputStream () :返回完整的请求体内容(包含每个表单项的部分)
自己来对getInputStream()方法返回的流进行分割,太麻烦了 可以使用工具来对整个表单数据进行分割处理。
===========================
1. 上传组件
* commons-fileupload
导jar包
* commons-fileupload.jar
>commons-io.jar
2. 核心类
* 工厂 : DiskFileItemFactory,用来创建解析器!
* 解析器 : ServletFileUpload,用来解析request(他内部的InputStream), 解析后会得到N个表单项
* 表单项 : FileItem,一个FileItem对象对应一个表单项,他可能是普通表单项,也可能是文件表单项!使用该类的IPA可以方便的操作表单数据。
3. 核心代码:
* 创建工厂:DiskFileItemFactory factory = new DiskFileItemFactory();
* 创建解析器:ServletFileUpload sfu = new ServletFileUpload(factory);
* 解析request,得到N个FileItem:List<FileItem> fileItemList = sfu.parseRequest(request);
4. FileItem 的API
* 普通表单项
>String getFieldName(); 获取表单项名称
>String getString (String encoding); 获取表单项的值,可以传递字符编码!
* 文件表单项
>String getFieldName(); 获取表单名称
>String getName() : 获取上传文件的名称(有的浏览器可能会包含客户端路径)
>String getContentType() : 获取上传文件的mime类型
> byte[] get() : 获取文件内容(字符数组形式)
> InputStream getInputStream(); 获取文件内容(流形式)
> long getSize() : 获取上传文件的字节数
> boolean isFormField(); 判断当前表单项是否为普通表单项,返回true表示是普通表单项,返回false表示为文件表单项
>void write(File file): 把上传文件保存到指定路径
> void delete(): 删除临时文件!
细节
1. 上传文件必须保存到WEB-INF 目录下
* 用户上传的是jsp,然后自己再去访问,就会执行用户的jsp
2. 文件名称相关细节,
* 文件路径问题: fileItem.getName() 可以获取上传文件的名称
>不同浏览器效果不同:
* 有的浏览器文件名称只是名称,没有路径!
* 有的浏览器文件名称中包含路径!
> 需要把包含路径的名称进行分割!
* 文件名称中文编码
> request.setCharacterEncoding("utf-8")
> servletFileUpload.setHeaderEncoding("utf-8"); 他的优先级高
* 文件同名问题
> 多人上传文件名称相同会出现覆盖问题
> 给文件名称添加uuid前缀,即可!
3. 目录打散问题
* 不让一个目录下存放过多的文件
* 打散方式:
> 时间打散:例如一天生成一个目录!
> 首字母打散:
> 哈希打散:
* 文件名称 ---》 哈希码(int hcode = filename.hashCode())
* 哈希码 ---》 16进制的哈希码 (String hstr = Integer.toHexString(hcode))
* 16 进制哈希码 ---》 获取第一个字母,使用它生成一个目录,生成一个a目录
* 16 进制哈希码 ---》 获取第一个字母,使用它生成一个目录(在第一个目录下生成)在a目录下生成一个b目录
* 把文件保存到/第一个字母/第二个字母/ 文件
1.通过文件名称得到hCode
int hCode = filename.hashCode();
2. 把hCode转换成16进制字符串
String hStr = Interger.toHexString(hCode);
3. 获取hCode第一个和第二个字母
char c1 = hStr.charAt(0);
char c2 = hStr.charAt(1);
把c1 和c2与savepth连接在一起
savepath = savepath + "/" + c1 + "/" +c2;
生成c1和c2目录
File path = new File(savepath);
path.mkdirs(); //创建目录链,如果目录已经存在,就不会重复创建!
生成目录文件的File对象
File destFile = new File(path, realname);
保存
fi.write(destFile);
4. 大小限制问题
* 单个文件大小限制
>servletFileUpload.setFileSizeMax(long)
>这个方法必须在解析之前调用!
>当上传的文件超出限制时,parseRequest()方法会抛出异常!
* 总大小限制(整个请求也就是整个表单)
>servletFileUpload.setSizeMax(long);
>这个方法必须在解析之前调用!
>当上传的文件超出限制时,parseRequest()方法会抛出异常!
5. 缓存大写与临时目录
* 客户端上传的文件是否保存到内存中,还是保存到硬盘上。
* 设置缓存大小: 例如缓存大小为10KB,那么上传的文件如果大小在10KB之内,就不用保存到硬盘上,如果超出10KB先保存到硬盘上。
* 临时目录: 当需要把文件保存到硬盘上时,需要一个保存临时文件的目录!