一、文件上传编程
基本原理:
1、在用户页面中添加上传输入项 (客端页面操作)
<input type="file" />
注意事项:
1) 必须为文件上传input 提供name属性,否则文件上传内容不会被表单提交
2) 表单的提交是post (get提交数据在url地址上显示,有长度限制)
3) 设置enctype=multipart /form-data 使得文件上传编码 ----- MIME编码格式
2、在服务器端编写文件上传程序
通过request.getInputStream分析文件上传原理
二、Apache commons-fileupload 使用
1.导入jar包
同时下载 commons-fileupload 和 commons-io 两个包 -------- fileupload依赖io包
2.编程实现
步骤一:获得DiskFileItemFactory 文件项工厂
步骤二:通过工厂 获得文件上传请求核心解析类 ServletFileUpload
步骤三:使用ServletFileUpload对request进行解析 ---- 获得很多个FileItem
步骤四:对每个FileItem进行操作 判断FileItem是不是普通字段 isFormField
3.代表普通字段FileItem
getFieldName(); ---- 获得表单项name属性
getString(); ----- 获得表单项value
4.代表文件上传FileItem
getInputStream() --- 获得文件内容输入流
getName() ------ 获得上传文件名称
三、commons-fileupload 核心API 分析
1.DiskFileItemFactory 磁盘文件项工厂类
构造工厂时,指定内存缓冲区大小和临时文件存放位置
public DiskFileItemFactory(int sizeThreshold, java.io.File repository)
设置内存缓冲区大小,默认10K
public void setSizeThreshold(int sizeThreshold)
设置临时文件存放位置,默认System.getProperty("java.io.tmpdir")
public void setRepository(java.io.File repository)
2.ServletFileUpload 文件上传核心类
判断request的编码方式是否为multipart/form-data
static boolean isMultipartContent(javax.servlet.http.HttpServletRequest request)
解析request,将请求体每个部分封装FileItem对象,返回List<FileItem>
java.util.List parseRequest(javax.servlet.http.HttpServletRequest request)
设置单个文件上传大小 和 void setSizeMax(long sizeMax) 设置总文件上传大小
void setFileSizeMax(long fileSizeMax)
设置编码集 解决上传文件名乱码 *****
void setHeaderEncoding(java.lang.String encoding)
设置文件上传监听器 (用来监控文件上传进度)
void setProgressListener(ProgressListener pListener)
3.FileItem 表示文件上传表单中 每个数据部分
判断该数据项是否为文件上传项,true 不是文件上传 false 是文件上传
boolean isFormField()
if(fileItem.isFormField()){
// 不是上传项
java.lang.String getFieldName() 获得普通表单项name属性
java.lang.String getString() / java.lang.String getString(java.lang.String encoding) 获得普通表单项value属性 传入编码集用来解决输入value乱码
}else{
// 是上传项
java.lang.String getName() 获得上传文件名 (注意IE6存在路径)
java.io.InputStream getInputStream() 获得上传文件内容输入流
// 上传文件
void delete() 删除临时文件(删除时,必须要管理输入输出流)
}
4.乱码问题:
普通编写项 value属性乱码 ------------- fileItem.getString(编码集);
上传文件项 文件名乱码 --------- fileupload.setHeaderEncoding(编码集);
四、上传文件的进度监控
ServletFileUpload 类 提供 public void setProgressListener(ProgressListener pListener)
1.为文件上传程序绑定一个监听器对象,通过监听器可以监听文件上传全过程
2.和AJAX技术结合,编写文件上传进度条
设置监听器,文件上传程序会自动执行 监听器中 update方法 public void update(long pBytesRead, long pContentLength, int pItems)
在方法中可以获得 文件总大小、已经上传大小和 上传第几个元素
能否根据上面三个参数计算:剩余大小、传输速度、已用时间、剩余时间
1) 已用时间 = 当前时间 - 开始时间
2) 速度 = 已经上传大小/已用时间
3) 剩余大小 = 总大小- 已经上传大小
4) 剩余时间 = 剩余大小/速度
五、实例
package com.jjyy.web;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.math.BigDecimal;
import java.util.List;
import java.util.UUID;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.ProgressListener;
import org.apache.commons.fileupload.FileUploadBase.FileSizeLimitExceededException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
import com.itheima.util.IOUtils;
public class UploadServlet extends HttpServlet {
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
try{
//1.创建工厂
DiskFileItemFactory factory = new DiskFileItemFactory();
factory.setSizeThreshold(100*1024);
factory.setRepository(new File(this.getServletContext().getRealPath("WEB-INF/temp")));
//2.生产文件上传核心类
ServletFileUpload fileUpload = new ServletFileUpload(factory);
//--检查是否是正确的文件上传表单
if(!fileUpload.isMultipartContent(request)){
throw new RuntimeException("请用正确的表单进行上传!");
}
//--设置文件上传的大小限制
// fileUpload.setFileSizeMax(1024*1024*100);//单个文件不大于10M
// fileUpload.setSizeMax(1024*1024*100);//总大小不大于100M
//--设置编码集,解决上传文件名的乱码问题
fileUpload.setHeaderEncoding("utf-8");
//--设置文件上传监听
fileUpload.setProgressListener(new ProgressListener(){
Long beginTime = System.currentTimeMillis();
public void update(long bytesRead, long contentLength, int items) {
BigDecimal br = new BigDecimal(bytesRead).divide(new BigDecimal(1024),2,BigDecimal.ROUND_HALF_UP);
BigDecimal cl = new BigDecimal(contentLength).divide(new BigDecimal(1024),2,BigDecimal.ROUND_HALF_UP);
System.out.print("当前读取的是第"+items+"个上传项,总大小"+cl+"KB,已经读取"+br+"KB");
//剩余字节数
BigDecimal ll = cl.subtract(br);
System.out.print("剩余"+ll+"KB");
//上传百分比
BigDecimal per = br.multiply(new BigDecimal(100)).divide(cl,2,BigDecimal.ROUND_HALF_UP);
System.out.print("已经完成"+per+"%");
//上传用时
Long nowTime = System.currentTimeMillis();
Long useTime = (nowTime - beginTime)/1000;
System.out.print("已经用时"+useTime+"秒");
//上传速度
BigDecimal speed = new BigDecimal(0);
if(useTime!=0){
speed = br.divide(new BigDecimal(useTime),2,BigDecimal.ROUND_HALF_UP);
}
System.out.print("上传速度为"+speed+"KB/S");
//大致剩余时间
BigDecimal ltime = new BigDecimal(0);
if(!speed.equals(new BigDecimal(0))){
ltime = ll.divide(speed,0,BigDecimal.ROUND_HALF_UP);
}
System.out.print("大致剩余时间为"+ltime+"秒");
System.out.println();
}
});
//3.利用文件上传核心类解析request
List<FileItem> list = fileUpload.parseRequest(request);
//4.遍历所有的FileItem
for(FileItem item : list){
if(item.isFormField()){
//当前是一个普通的字段项
String name = item.getFieldName();
String value = item.getString("utf-8");
System.out.println(name+":"+value);
}else{
//当前是一个文件上传项
String filename = item.getName();
String uuidName = UUID.randomUUID().toString()+"_"+filename;
int hash = uuidName.hashCode();
String hashStr = Integer.toHexString(hash);
char [] hss = hashStr.toCharArray();
String path = this.getServletContext().getRealPath("WEB-INF/upload");
for(char c : hss){
path+="/"+c;
}
new File(path).mkdirs();
InputStream in = item.getInputStream();
OutputStream out = new FileOutputStream(new File(path,uuidName));
IOUtils.In2Out(in, out);
IOUtils.close(in, out);
//--删除临时文件
item.delete();
}
}
}catch (FileSizeLimitExceededException e) {
response.getWriter().write("单个文件不超过10M,总大小不超过100M!");
return;
}catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doGet(request, response);
}
}