JavaWeb 之文件上传
1. 文件上传的要求
1.1 上传对表单的限制
method="post";enctype="multipart/form-data";- 表单中需要添加文件表单项:
<input type="file" name="xxx"/>.
1.2 上传对 Servlet 的限制
request.getParameter("xxx")方法返回的是字符串类型, 所以在表单中有
enctype="multipart/form-data"时, 该方法作废了, 因为它永远都返回 null.- 应该使用
ServletInputStream request.getInputStream(), 返回结果中包含整个请求体. - 上传不能使用 BaseServlet,因为BaseServlet 内部调用了
getParameter()方法.
2. 多部件表单体

- 一个表单被分割出多个部件,即一个表单项一个部件;
- 一个部件中自己包含请求头和空行, 以及请求体;
- 普通表单项
- 包含一个请求头:
Content-Disposition:xxx; name="表单项名称"; - 请求体就是表单项的值.
- 包含一个请求头:
- 文件表单项
- 包含两个请求头:
Content-Disposition:xxx; name="表单项名称"; filename="上传文件的名称";Content-Type: 上传文件的 MIME 类型;- 请求体就是上传文件的内容.
3. 相关 jar 包
commons-fileupload.jarcommons-io.jar- 这个组件会解析 request 中的上传数据, 解析后的结果是,一个表单项数据封装到一个 FileItem 对象中.
我们只需要调用 FileItem 的方法即可.
4. 上传三步
4.1 上传涉及的相关类
- 工厂类:
DiskFileItemFactory; - 解析器类:
ServletFileUpload; - 表单项类:
FileItem;
4.2 具体步骤
- 创建工厂:
DiskFileItemFactory factory = new DiskFileItemFactory(); - 创建解析器:
ServletFileUpload sfu = new ServletFileUpload(factory); - 使用解析器来解析 request, 得到FileItem集合:
List<FileItem> fileItemList = sfu.parseRequest(request);
4.3 FileItem 对象中的方法 (commons-fileupload API)
boolean isFormField(): 是否为普通表单项. true,表示为普通表单项; false,表示为文件表单项;String getFieldName(): 返回当前表单项的名称;String getString(String charset): 返回表单项的值, charset 默认值为 "utf-8";String getName(): 返回上传文件的名称;long getSize(): 返回上传文件的字节数;InputStream getInputStream(): 返回上传文件对应的输入流;void write(File destFile): 把上传文件的内容保存到指定的文件中;String getContentType(): 获取上传文件的 MIME 类型;
5. 上传的细节
5.1 文件必须保存到 WEB-INF 下!
- 目的是不让浏览器直接访问到.
5.2 文件名称相关问题
- IE6 浏览器上传的文件名称是绝对路径(包含磁盘的路径),需要将磁盘部分切割, 例如: "c:\files\a.jpg";
- 文件名乱码或普通表单项乱码:
request.setCharacterEncoding("utf-8"), fileupload 内部会调用request.getCharacterEncoding()方法;servletFileUpload.setHeaderEncoding("utf-8"), 这种方式的优先级高于前一种.
- 文件同名问题: 需要为每个文件添加名称前缀, 为了保证不重复, 可以使用 uuid
filename = CommonUtils.uuid()+"_"+filename;
5.3 目录打散
- 不能在一个目录下存放过多文件
- 首字母打散: 使用文件的首字母作为目录名称; 不方便操作中文的文件名
- 时间打散: 使用当前日期作为目录;
- 哈希打散:
- 通过文件名称获得 int 值, 即调用
hashCode(); - 把 int 值转换成十六进制 "0~9 和 A ~ F";
- 获取十六进制的前两位用来生成目录, 目录为两层! 例如: "1B2C3D4E5F", /1/B 保存文件.
- 通过文件名称获得 int 值, 即调用
5.4 上传文件的大小限制
单个文件的大小限制
sfu.setFileSizeMax(100 * 1024): 表示限制单个文件大小为 100K;- 必须在
parseRequest()方法之前调用; - 如果上传的文件超出限制, 在 parseRequest() 方法执行时, 会抛出异常!!
FileUploadBase.FileSizeLimitExceedeException.
整个请求所有数据大小限制
sfu.setSizeMax(1024 * 1024): 表示限制整个表单大小为 1M;- 必须在
parseRequest()方法之前调用; - 如果上传的文件超出限制, 在 parseRequest() 方法执行时, 会抛出异常!!
FileUploadBase.SizeLimitExceededException.
5.5 缓存大小与临时目录
- 缓存大小: 上传文件超出多大时, 才向硬盘保存! 默认 10KB;
- 临时目录: 向硬盘的什么目录保存;
- 设置缓存大小和临时目录:
new DiskFileItemFactory(20 * 1024, new File("F:/temp"))
// 目录打散
public void UploadServlet extends HttpServlet{
public void doPost(HttpServletRequest request, HttpServletResponse resp)
throws ServletException, IOException{
request.setCharacterEncoding("UTF-8");
response.setContentType("text/html;utf-8");
// 文件上传三步
DiskFileItemFactory factory = new DiskFileItemFactory();
ServletFileUpload sfu = new ServletFileUpload(factory);
try{
List<FileItem> fileItemList = sfu.parseRequest(request);
// 获取照片文件表单项
FileItem fi = fileItemList.get(1);
// 得到保存上传文件的根路径
String root = this.getServletContext().getRealPath("/WEB-INF/files/");
// 得到文件名
String filename = fi.getName();
//处理文件名的绝对路径问题
int index = filename.lastIndexOf("\\");
if(index != -1){
filename = filename.substring(index+1);
}
// 给文件名添加 uuid 前缀, 处理文件同名问题
String savename = CommonUtils.uuid()+"_"+filename;
// 得到文件名的 hashCode, 生成两层目录
int hCode = filename.hashCode();
String hex = Integer.toHexString(hCode);
File dirFile = new File(root,hex.charAt(0)+"/"+hex.charAt(1));
// 如果目录不存在, 创建目录链
dirFile.mkdirs();
// 创建目标文件
File destFile = new File(dirFile,savename);
// 保存
fi.save(destFile);
}catch(Exception e){
throw new RuntimeException(e);
}
}
}
参考资料:
JavaWeb 之文件上传的更多相关文章
- JavaWeb:实现文件上传
JavaWeb:实现文件上传 理解文件上传: 1.上传文件就是把客户端的文件发送给服务器端. 2.HTTP响应的正文部分最常见的是HTML文档,但是也可以是其他任意格式的数据,如图片和声音文件中的数据 ...
- JavaWeb实现文件上传下载功能实例解析
转:http://www.cnblogs.com/xdp-gacl/p/4200090.html JavaWeb实现文件上传下载功能实例解析 在Web应用系统开发中,文件上传和下载功能是非常常用的功能 ...
- JavaWeb:实现文件上传与下载
JavaWeb:实现文件上传与下载 文件上传前端处理 本模块使用到的前端Ajax库为Axio,其地址为GitHub官网. 关于文件上传 上传文件就是把客户端的文件发送给服务器端. 在常见情况(不包含文 ...
- JavaWeb实现文件上传下载功能实例解析 (好用)
转: JavaWeb实现文件上传下载功能实例解析 转:http://www.cnblogs.com/xdp-gacl/p/4200090.html JavaWeb实现文件上传下载功能实例解析 在Web ...
- 使用传统javaweb进行文件上传
使用传统文件上传方式 1.配置依赖 <properties> <project.build.sourceEncoding>UTF-8</project.build.sou ...
- JavaWeb多文件上传及zip打包下载
项目中经常会使用到文件上传及下载的功能.本篇文章总结场景在JavaWeb环境下,多文件上传及批量打包下载功能,包括前台及后台部分. 首先明确一点: 无法通过页面的无刷新ajax请求,直接发下载.上 ...
- 深入分析JavaWeb Item40 -- 文件上传和下载
在Web应用系统开发中,文件上传和下载功能是很经常使用的功能,今天来讲一下JavaWeb中的文件上传和下载功能的实现. 对于文件上传.浏览器在上传的过程中是将文件以流的形式提交到server端的.假设 ...
- javaweb大文件上传
本文主要关于利用html表单上传文件的后台代码实现. 需要用到两个工具类Apache commons-fileupload和commons-io. 注意要校验是否选择文件上传,最开始写的时候没有加上校 ...
- javaWeb实现文件上传与下载 (转)
文件上传概述 实现web开发中的文件上传功能,需完成如下二步操作: 在web页面中添加上传输入项 在servlet中读取上传文件的数据,并保存到本地硬盘中. 如何在web页面中添加上传输入项? < ...
随机推荐
- 在Windows Python3.5 安装LightGBM
LightGBM是微软旗下DMTK推出的Gradient Boosting框架,因为其快速高效,以后或许会成为数据挖掘竞赛中的又一个大杀器.地址:https://github.com/Microsof ...
- unity, inspector debug
以前经常因为脚本中private变量不在inspector界面中显示,不方便观察其值的变化,所以本该用private的用了public. 今天发现,原来inspector有个选项,如图,平常勾选的是N ...
- CAN总线过滤规则
奇怪的设计 If (AFMR & Message_ID) == (AFMR & AFIR) then Capture Message AFIR 1 接收 AFMR 0 接收 设置某接收 ...
- layui表格渲染中模板的使用举例
实例一: { field: 'status', align: 'center', title: '活动状态', templet: function (d) { if (d.status == &quo ...
- 服务器上nginx反向代理的配置
Nginx不但是一款高性能的Web服务器,也是高性能的反向代理服务器.下面简单说说Nginx的反向代理功能. 反向代理是什么? 反向代理指以代理服务器来接受Internet上的连接请求,然后将请求转发 ...
- 如何获取wifi名称(SSID)
@import SystemConfiguration.CaptiveNetwork; /** Returns first non-empty SSID network info dictionary ...
- 华为终端-新浪微博联合创新,3D建模+AR 成就全新社交体验
近日,全球首款搭载3D感知摄像头的手机华为Mate 20发布. 通过Mate 20自带的景深摄像头及麒麟980的NPU加速能力,手机能够在获取物体表面信息后,完成高速的精细化3D建模. 那么,如何让3 ...
- D - Sigma Function 1~n内有多少个约数和为偶数
/** 题目:D - Sigma Function 链接:https://vjudge.net/contest/154246#problem/D 题意:求1~n内约数和为偶数的数的个数. 思路:一个数 ...
- iOS图片加水印效果的实现并保存至相冊
图片加水印效果的实现并保存至相冊 实现效果如图: project下载:githubproject下载链接 代码: - (void)viewDidLoad { [super viewDidLoad]; ...
- C++之类的静态成员变量和静态成员函数
static静态成员函数 在类中.static 除了声明静态成员变量,还能够声明静态成员函数. 普通成员函数能够訪问全部成员变量.而静态成员函数仅仅能訪问静态成员变量. 我们知道.当调用一个对象的成员 ...