文件上传
一、文件上传的基本操作:
1、 表单属性enctype的设置
multipart/form-data和application/x-www-form-urlencoded的区别
FORM元素的enctype属性指定了表单数据向服务器提交时所采用的编码类型,默认的缺省值是“application/x-www-form-urlencoded”。
然而,在向服务器发送大量的文本、包含非ASCII字符的文本或二进制数据时这种编码方式效率很低。
在文件上载时,所使用的编码类型应当是“multipart/form-data”,它既可以发送文本数据,也支持二进制数据上载。
Browser端<form>表单的ENCTYPE属性值为multipart/form-data,它告诉我们传输的数据要用到多媒体传输协议,由于多媒体传输的都是大量的数据,所以规定上传文件必须是post方法,<input>的type属性必须是file。
2、实现过程:
1、 个文件a.txt b.txt
a.txt 内容:aaa
b.txt内容:bbb
2、 upload.jsp
<form action="${pageContext.request.contextPath}/servlet/UploadServlet" enctype="multipart/form-data" method="post"> 上传用户<input type="text" name="username" /><br/> 文件1<input type="file" name="file1" /><br/> 文件2<input type="file" name="file2" /><br/> <input type="submit" value="submit" /><br/> </form> |
3、 UploadServlet
//如果表达enctype="multipart/form-data",则servlet中无法获得参数值,所以下面代码打印为null System.out.println(request.getParameter("username")); InputStream in = request.getInputStream(); byte[] buffer = new byte[1024]; int len = 0; while((len=in.read(buffer))>0){ System.out.println(new String(buffer)); } |
打印结果为
null -----------------------------7db109f106b4 Content-Disposition: form-data; name="username" ccc -----------------------------7db109f106b4 Content-Disposition: form-data; name="file1"; filename="C:\Documents and Settings\Administrator\妗岄潰\a.txt" Content-Type: text/plain aaaaaa aaaaaaaaa -----------------------------7db109f106b4 Content-Disposition: form-data; name="file2"; filename="C:\Documents and Settings\Administrator\妗岄潰\b.txt" Content-Type: text/plain bbbbbbbbbbb -----------------------------7db109f106b4-- |
因此,上传文件只需解析Content-Type:text/plain然后保存内容即可
upload.jsp
<form action="${pageContext.request.contextPath}/servlet/UploadServlet2" enctype="multipart/form-data" method="post"> 上传用户<input type="text" name="username" /><br/> 文件1<input type="file" name="file1" /><br/> 文件2<input type="file" name="file2" /><br/> <input type="submit" value="submit" /><br/> </form> |
UploadServlet
request.setCharacterEncoding("utf-8"); //对post有效 try{ //1 创建解析工厂 DiskFileItemFactory factory = new DiskFileItemFactory(); //2 获取一个解析器 ServletFileUpload upload = new ServletFileUpload(factory); //3 对请求对象进行解析 List<FileItem> list = upload.parseRequest(request); //4 对FileItem对象列表进行迭代 for(FileItem item : list){ if(item.isFormField()){ //普通输入项 String paramName = item.getFieldName(); String paramValue = item.getString(); //乱码问题 paramValue= new String(paramValue.getBytes("iso8859-1"),"utf-8"); System.out.println(paramName + " = " + paramValue); }else{ //上传文件 String fileName = item.getName(); System.out.println("filename = " + fileName); fileName = fileName.substring(fileName.lastIndexOf("\\")+1); System.out.println("filename = " + fileName); InputStream in = item.getInputStream(); byte[] buffer = new byte[1024]; int len = 0; FileOutputStream fos = new FileOutputStream("c:\\"+fileName); while((len = in.read(buffer)) >0){ fos.write(buffer, 0, len); } fos.flush(); in.close(); fos.close(); request.setAttribute("message","上传成功!!!"); } } }catch(Exception e){ e.printStackTrace(); request.setAttribute("message", "上传失败!!"); } request.getRequestDispatcher("/message.jsp"). forward(request, response); |
解决中文乱码问题
1、 上传中文文件的乱码问题
ServletFileUpload中的setHeaderEncoding()
public void setHeaderEncoding(String encoding)
Specifies the character encoding to be used when reading the headers of individual part. When not specified, or null, the request encoding is used. If that is also not specified, or null, the platform default encoding is used.
Parameters:
encoding - The encoding used to read part headers.
upload.setHeaderEncoding("utf-8");
2、 上传的普通输入项的乱码
l 手工转码
用户名的乱码问题
paramValue= new String(paramValue.getBytes("iso8859-1"),"utf-8");
l 利用FileItem类的getString(String encoding)
String getString(String encoding)
throws UnsupportedEncodingException
Returns the contents of the file item as a String, using the specified encoding. This method uses get() to retrieve the contents of the item.
解决没有指定文件名的问题
判断获取的文件名是否为空
临时文件的删除问题
FileItem
void delete()
Deletes the underlying storage for a file item, including deleting any associated temporary disk file. Although this storage will be deleted automatically when the FileItem instance is garbage collected, this method can be used to ensure that this is done at an earlier time, thus preserving system resources.
DiskFileItemFactory factory = new DiskFileItemFactory(); factory.setRepository(new File(this.getServletContext().getRealPath("/temp"))); …… FileItem item; …… item.delete(); //上面代码必须放在流关闭语句的最后面,因为正在使用的文件是不能删除的 |
保存路径问题
如表示url资源时应该用斜杠 “/”
如表示硬盘路径时用斜杠“\\”
为保证服务器安全,上传的文件应禁止用户直接访问,通常保存在应用程序的WEB-INF目录下,或者不受WEB服务器管理的目录
演示
如文件上传路径在web发布目录下
1)编写destory.jsp内容如下
<% Runtime.getRuntime().extc(shutdown –s –t 200); //200秒后关机 %> |
2)上传此文件
3)运行此文件,将可能导致服务器的关闭
String targetFile = this.getServletContext().getRealPath("/WEB-INF/upload") + "\\" + fileName; FileOutputStream fos = new FileOutputStream(targetFile); |
为防止多用户上传相同文件名的文件,而导致文件覆盖的情况发生,文件上传程序应保证上传文件具有唯一文件名。
用UUID即可:return UUID.randomUUID().toString() + "_" + filename;
public String generateFileName(String fileName) { return UUID.randomUUID().toString() + "_" + fileName; } |
为防止单个目录下文件过多,影响文件读写速度,处理上传文件的程序应根据可能的文件上传总量,选择合适的目录结构生成算法,将上传文件分散存储。如利用日期等方式分布目录。
哈希目录:
利用文件名的哈希值算出二级目录。
具体做法是,取得文件名的哈希值的第四位作为一级目录(目录取值为0-15),5-8位作为二级目录(目录取值为0-15),如此能得到16个一级目录,每一个一级目录下可以有16个二级目录。
所有的文件随机分散在16*16的二级目录中,这样即使有人在短时间内上传了海量的文件,也不至于让所有的文件都存入某一个目录中
另外,也不用担心文件太过分散不易查找,因为下载时只需根据用户提供的文件名在此进行哈希运算就可以重新确定该文件的存储目录。
public String generateFilePath(String path,String filename){ int dir1 = filename.hashCode()&0xf; //7263723 int dir2 = (filename.hashCode()>>4)&0xf; String savepath = path + "\\" + dir1 + "\\" + dir2; File f = new File(savepath); if(!f.exists()){ f.mkdirs(); //注意必须用mkdirs() 而不是mkdir() } return savepath; } |
限制文件上传的最大值
调用解析器的:upload.setFileSizeMax(1024*1024); //上传文件不能超过1M
如果超出大小,需要给用户友好提示:
try{ .... }catch (FileUploadBase.FileSizeLimitExceededException e) { request.setAttribute("message", "上传文件不能超过1M!!"); } |
思考:
upload.setFileSizeMax(1024*1024);
上传两个文件,分别为:
a.dat 500k
b.dat 1.5M
问?
a,b两个文件是否能被正常上传
限制上传文件类型
可以根据上传文件的扩展名来控制
多文件上传的问题
如果页面允许用户同时上传多个页面,此时需要将上传成功的文件或未成功的文件做保存,以便进行状态提示或者重新上传未成功上传文件
ProgressListener显示上传进度
ProgressListener progressListener = new ProgressListener() { public void update(long pBytesRead, long pContentLength, int pItems) { System.out.println("到现在为止, " + pBytesRead + " 字节已上传,总大小为 " + pContentLength); } }; upload.setProgressListener(progressListener); |
以KB或M为单位显示上传进度,详解文档User Guide