文件上传是开发中常用的功能,本节主要介绍用commons-fileupload-1.1.jar包实现基本的文件上传功能,即上传文件到指定的目录中,同时介绍上传过程中使用到的相关类及其方法。
下载插件:
准备需要的jar包
* commons-fileupload-1.1.jar 文件上传jar包(必须导入)
* commons-io-1.2.jar (必须导入)如果不导入的程序编译时不会报错,但是发布后运行时会报错
* log4j-1.2.8.jar 强烈建议导入,但在本类中不是必须的,后面的上传文件到数据库中的类中会使用
* classes12.jar 连接oracle数据库的jar包,后面的上传文件到数据库中时必须导入
开发环境:
System:xp JDK:1.5(开发的JDK版本) Tomcat:5.X Myeclipse:6.5
注意:编译的JDK版本用JDK1.4(如果用JDK1.5编译,会报”Unsupported major.minor version 49.0”错误)
项目环境:
文件上传目录介绍:
在D盘根目录新建一个test文件夹,然后在test文件夹中新建以下两个文件夹
* temp:存放超过设置大小的文件(>2M的文件)
* uploadfile:上传文件的存放目录(<=2M的文件)
class&method:
*DiskFileItemFactory
* 实现了FileItemFactory接口
* 主要方法有:
* public FileItem createItem(String fieldName, String contentType, boolean isFormField, String fileName)
* setRepository(File repository);设置缓存路径 也有相应的get方法
* setSizeThreshold(int sizeThreshold);设置上传文件内存缓冲区的大小 也有相应的get方法
*
* ServletFileUpload
* 从FileUpload继承,而FileUpload又从FileUploadBase继承
* ServletFileUpload(FileItemFactory fileItemFactory);传入一个fileItemFactory对象,构造ServletFileUpload对象
* List parseRequest(HttpServletRequest request);从request中获得文件请求列表
* void setSizeMax(long sizeMax);//设置允许上传文件大小 也有相应的get方法
* setHeaderEncoding(String encoding);//设置上传文件名编码的格式 也有相应的get方法
*
* Pattern
* 正则表达式的编译表示形式,指定为字符串的正则表达式必须首先被编译为此类的实例。
* 然后,可将得到的模式用于创建 Matcher 对象,依照正则表达式,该对象可以与任意字符序列匹配
* static Pattern compile(String regex);将给定的正则表达式编译到模式中
* Matcher matcher(CharSequence input);创建匹配给定输入与此模式的匹配器,返回此模式的新匹配器
*
* FileItem
* 抽象接口,继承Serializable
* boolean isFormField();判断FileItem类对象封装的数据是否属于一个普通表单字段,
* 还是属于一个文件表单字段,如果是普通表单字段则返回true,否则返回false
* String getName();用于获得文件上传字段中的文件名
* String getFieldName();用于返回表单字段元素的name属性值
* long getSize();获得上传文件的大小 单位:字节
* getContentType();用于获得上传文件的类型
* void write(File paramFile);用于将FileItem对象中保存的主体内容保存到某个指定的文件中
*
* Matcher
* 通过解释 Pattern 对 character sequence 执行匹配操作的引擎。
* 通过调用模式的 matcher 方法从模式创建匹配器。创建匹配器后,可以使用它执行三种不同的匹配操作:
* matches方法尝试将整个输入序列与该模式匹配。
* lookingAt 尝试将输入序列从头开始与该模式匹配。
* find 方法扫描输入序列以查找与该模式匹配的下一个子序列。
* String group(int group)返回在以前匹配操作期间由给定组捕获的输入子序列
*
* String
* boolean endsWith(String suffix)测试此字符串是否以指定的后缀结束。
start
1.新建web项目UpDown,导入上面介绍的4个jar包
2.新建upload.html文件,用于上传的前台的UI界面
upload.html
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<title>文件上传</title>
<meta name="Generator" content="EditPlus">
<meta name="Author" content="">
<meta name="Keywords" content="">
<meta name="Description" content="">
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
</head>
<body>
<h1>
文件上传演示demo---上传到本机文件夹中(程序和上传目录在同一主机)
</h1>
<!--
1.上传的内容有图片的时候,form中必须加入 ENCTYPE="multipart/form-data"
2.在<input>标签中必须有name属性(除了提交按钮submit),否则运行时虽然不会报错,但是文件无法成功上传
-->
<form name="uploadform" method="POST" action="upload"
ENCTYPE="multipart/form-data">
<table border="1" width="450" cellpadding="4" cellspacing="2"
bordercolor="#9BD7FF">
<tr>
<td width="100%" colspan="2">
文件1:
<input name="a" size="40" type="file">
</td>
</tr>
<tr>
<td width="100%" colspan="2">
文件2:
<input name=b” " size="40" type="file">
</td>
</tr>
<tr>
<td width="100%" colspan="2">
文件3:
<input name="c" size="40" type="file">
</td>
</tr>
</table>
<br />
<br />
<table>
<tr>
<td align="center">
<input name="upload" type="submit" value="开始上传" />
</td>
</tr>
</table>
</form>
</body>
</html>
3.新建servlet文件Upload.java文件,用于对上传的文件进行处理
package upload;
import java.io.File;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Iterator;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
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.FileUploadException;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
/**此程序是上传文件到指定的目录的一个小小的demo
* 有以下两个要求:(1)上传的文件不能超过文件指定的最大值(2)对有些后缀名的文件不能上传
*/
/**
*Module: Upload.java
*Description: 上传文件到指定的目录(程序和上传的目录在同一台主机)
*Company: XXX
*Author: ptp
*Date: Feb 6, 2012
*/
public class Upload extends HttpServlet {
// 定义常量,保存文件路径
private static final String FILE_PATH = "D:" + File.separator + "test"
+ File.separator + "upload" + File.separator;// 文件上传的路径
private static final String FILE_TEMP = "D:" + File.separator + "test"
+ File.separator + "temp" + File.separator;;// 文件缓存路径
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html; charset=GB2312");
PrintWriter out = response.getWriter();
// ServletFileUpload.isMultipartContent(request);
// 可以处理之前用上面的方法检测request中是否有multipart内容,不过,已经是废弃的方法了
// 生成DiskFileItemFactory工厂
DiskFileItemFactory factory = new DiskFileItemFactory();
// 对工厂进行相关的配置
// 设置最多只允许在内存中存储的数据,单位:字节
factory.setSizeThreshold(2048);
// 设置一旦文件大小超过getSizeThreshold()的值时数据存放在硬盘的目录 文件缓存路径 【暂时没有实现这个功能】
//判断指定的目录是否存在,如果不存在则新建该目录,注意mkdirs()和mkdir()的区别:
//如果test不存在,用mkdir()程序会在后面会报错,用mkdirs()就不会报错
File fileTemp = new File(FILE_TEMP);
if (!fileTemp.exists()){
fileTemp.mkdirs();
}
File filePath1 = new File(FILE_PATH);
if (!filePath1.exists()) {
filePath1.mkdir();//此处可以用mkdir()方法,因为前面的代码执行后test目录一定存在
}
//设置缓存路径
factory.setRepository(fileTemp);
// 将DiskFileItemFactory对象传给ServletFileUpload构造方法,生成上传类ServletFileUpload的对象
ServletFileUpload sevletFileUpload = new ServletFileUpload(factory);
// 设置允许用户上传文件大小,单位:字节,这里设为2M
sevletFileUpload.setSizeMax(2 * 1024 * 1024);
// 设置编码,解决上传文件名乱码
sevletFileUpload.setHeaderEncoding("GB2312");/**按照JDK的介绍,此方法是解决上传文件名中文乱码问题,但是后面测试发现不能实现*/
//获得允许用户上传文件大小
long maxSize = sevletFileUpload.getSizeMax();
try {
/**
* 开始读取上传信息 从request中取到上传文件列表
* 超过了设置的最大值2M会报异常
* 文件缓存目录不存在也会报异常(前面已经对目录存在做了判断,所以不可能出现目录不存在报的情况)
*/
List fileItems = sevletFileUpload.parseRequest(request);
//调用上传文件的方法
if (null != fileItems)
upLoadFile(fileItems, out);
} catch (FileUploadException e) {
out.println("对不起,您上传的文件大小超过了允许的最大值" + maxSize / 1024 / 1024 + "M!");
e.printStackTrace();
}
}
/**
* 上传文件
* @param fileItems
* @param out
*/
public void upLoadFile(List fileItems,PrintWriter out){
try{
// 依次处理每个上传的文件
Iterator iter = fileItems.iterator();
// 正则匹配,过滤路径取文件名
/*
* $ 行的结尾 . 任何字符(与行结束符可能匹配也可能不匹配) + 一次或多次
* \\\\匹配FILE_PATH中的\\,4个右斜杠转义后变为2个
*/
String regExp = ".+\\\\(.+)$";
// 过滤掉的文件类型
String[] errorType = { ".exe", ".com", ".cgi", ".asp" };
StringBuffer error=new StringBuffer();
//拼接不能上传的文件类型,即字符串error,用于后面显示
for(int i=0;i<errorType.length;i++){
if (i == errorType.length - 1) {
error.append(errorType[i] + "。");
} else {
error.append(errorType[i] + ",");
}
}
Pattern p = Pattern.compile(regExp);//将给定的正则表达式regExp编译到模式中
while (iter.hasNext()) {
FileItem item = (FileItem) iter.next();
/*忽略其他不是文件域的所有表单信息
*isFormField()方法判断FileItem类对象封装的数据是否属于一个普通表单字段,
*还是属于一个文件表单字段,如果是普通表单字段则返回true,否则返回false
*只有是文件表单字段才能进入if中的代码
*/
if (!item.isFormField()) {
String name = item.getName();// 用于获得上传文件的名称,如:c:\documents and settings\administrator\桌面\test.txt
long size = item.getSize();// 获得上传文件的大小
if ((name == null || name.equals("")) && size == 0)// 文件表单为空或文件大小为0跳过本次循环,继续下一次循环
continue;
Matcher m = p.matcher(name);// 创建匹配给定输入与此模式的匹配器
boolean result = m.find();// 方法扫描输入序列以查找与该模式匹配的下一个子序列
if (result) {
for (int temp = 0; temp < errorType.length; temp++) {
if (m.group(1).endsWith(errorType[temp])) {
throw new IOException("非法文件类型禁止上传,以下文件类型不能上传:"+error);
}
}
try {
//保存上传的文件到指定的目录
File filePath=new File(FILE_PATH + m.group(1));
item.write(filePath);
//在页面上输出上传文件后的信息
out.print(name + " " + size + "<br>");
} catch (Exception e) {
out.println(e);
}
} else {
throw new IOException("上传文件失败");
}
}
}
} catch (IOException e) {
out.println(e);
}
}
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doPost(request, response);
}
}
4.配置web.xml文件
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<servlet>
<servlet-name>upload</servlet-name>
<servlet-class>upload.Upload</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>upload</servlet-name>
<url-pattern>/upload</url-pattern>
</servlet-mapping>
</web-app>
5.发布项目
result
访问的URL如下:
http://localhost:8080/UpDown/upload.html
upload.html页面如下:
上传完成后跳转显示的页面如下:
检查指定的上传目录:
检查文件大小限制(上传一个大于2M的文件,然后点击”开始上传”,界面显示如下):
检查文件后缀名过滤(上传一个exe文件,然后点击”开始上传”,界面显示如下):
遗留&拓展部分:
* 1.中文乱码问题 IE界面显示乱码、上传目录乱码
* 2.上传目录和程序不在同一台主机上的处理方式
* 2.文件缓存问题
* 3.文件的端点续传功能
* 4.上传滚动条问题
answer
1.关于中文乱码问题的解决方法
经过测试以下语句设置为utf-8即可解决,设置为gbk或gb2312都无法解决中文乱码问题
sevletFileUpload.setHeaderEncoding("utf-8");