https://blog.csdn.net/luckyboyguo/article/details/12513079
一般情况下提交表单元素form
,默认的传输类型为application/x-www-form-urlencoded,后台可以通过request.getParameter()
获取String类型的表单值。但是当iuput元素中有文件时,就不能通过这种方式传递。
1. 文件上传的条件
- 表单必须是post提交方式
- 表单中必须有文件上传项,文件上传必须有name属性和值
- 表单中的enctype属性必须设置为multipart/form-data
<form method="POST" enctype="multipart/form-data" action="fup.cgi">
File to upload: <input type="file" name="upfile"><br/>
Notes about the file: <input type="text" name="note"><br/>
<br/>
<input type="submit" value="Press"> to upload the file!
</form>
- enctype=application/x-www-form-urlencoded 传递时,可以使用
request.getParameter()
获取 - enctype=multipart/form-data传递时,不可以使用
request.getParameter()
获取
2. Servlet获取上传文件
以form/data
方式传递的请求报文的body为:
------WebKitFormBoundaryJAjpF3viDTBKnfAy
Content-Disposition: form-data; name="username"
loserwang
------WebKitFormBoundaryJAjpF3viDTBKnfAy
Content-Disposition: form-data; name="password"
ddd
------WebKitFormBoundaryJAjpF3viDTBKnfAy
Content-Disposition: form-data; name="nickname"
ddd
------WebKitFormBoundaryJAjpF3viDTBKnfAy
Content-Disposition: form-data; name="sex"
male
------WebKitFormBoundaryJAjpF3viDTBKnfAy
Content-Disposition: form-data; name="picture"; filename="loserwang.jpg"
Content-Type: image/jpeg
������JFIF�����,�,�����C���
��
�
�������!����'#))'#&%,1?5,.;/%&6J7;ACFGF*4MRLDR?EFC���C���� �� C-&-CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC���v�����������������������������������������������������������������������������������������������������������������������������������������������������������
理论上在Servlet处可以获取request的body,分析结果,将上传文件的二进制写入本地,但是这样过于繁琐。最好的方式是使用第三方的jar包获取数据,这方面有很多现成的成熟优秀的jar包,比如commons-fileupload.jar。
3. commons-fileupload
Commons是Apache开发源代码组织的一个Java子项目,该项目主要涉及一些开发中常用的模块,如文件上传、命令行处理、数据库连接池等等
The Commons FileUpload package makes it easy to add robust, high-performance, file upload capability to your servlets and web applications.
3.1 步骤
- (1).下载commons-fileupload.jar和commons-io.jar并放到/Web/WEB-INF/lib。
(/Web/WEB-INF/lib 通常使用的jar文件)
- (2).读取文件
// 1.创建一个磁盘文件项工厂对象
DiskFileItemFactory diskFileItemFactory = new DiskFileItemFactory();
//2.创建一个核心解析类
ServletFileUpload servletFileUpload = new ServletFileUpload(diskFileItemFactory);
//3. 解析request请求,返回的是List集合,List集合中存放的是FileItem对象
try {
List<FileItem> list = servletFileUpload.parseRequest(request);
// 4.遍历集合,获得每个FileItem,判断是表单项还是文件上传项
for(FileItem fileItem:list){
//判断是表单项还是文件上传项
if(fileItem.isFormField()){
//点单项处理
}else
//文件上传项处理
}
}
3.2 表单项处理
假设存在一个多选项name = hobby
//遍历前的步骤
Map<String, String> map = new HashMap<String, String>();
List<String> hobbyList = new ArrayList<String>();
......
//读取表单数据
String name = fileItem.getFieldName();
String value = fileItem.getString("UTF-8");
//处理多选项。复选框为list
if("hobby" == name){
hobbyList.add(value);
String hobbyValue = hobbyList.toString().substring(1,hobbyList.size()-1);
map.put("hobby", hobbyValue);
}else{
//将结果存入map
map.put(name, value);
}
3.2 文件项处理
把上传文件放到 /web/upload:
//获取文件名
String fileName = fileItem.getFieldName();
//获取文件上传的数据
InputStream is = fileItem.getInputStream();
//获得文件路径
String path = this.getServletContext().getRealPath("/upload");
String url = path + "\\" + fileName ;
//将输入流对接到输出流
OutputStream os = new FileOutputStream(url);
int len = 0;
byte[] b = new byte[1024];
while ((len = is.read(b)) != -1){
os.write(b, 0, len);
}
is.close();
os.close();
这种情况虽然可以满足基本要求,但是如果上传的文件名相同,新文件会覆盖旧文件。为了每个文件不被覆盖,给每一个文件一个唯一的文件名。可以通过UUID来实现
public class UploadUtils {
/**
* 产生随机的文件名
*/
public static String getUUIDFileName(String fileName){
// 将文件名的前面部分进行截取: xx.jpg -> .jpg
int idx = fileName.lastIndexOf(".");
String extention = fileName.substring(0, idx);
//JAVA的UUID能产生随机的id
String uuidFileName = UUID.randomUUID().toString().replace("-","_") + extention;
return uuidFileName;
}
}
(IDEA踩坑:IDEA不编译空文件夹。/web/upload
一开始没有图片,所以不会在out
中产生upload文件夹,就会报错找不到路径
。我们直接移动一个图片到文件夹,选中Build-Rebuild Project
重新加载项目后,重启服务器 或者update resources即可)。