?
?

jsp文件上傳大多采用采用開源項目來簡化處理,這里列出常用的兩個jar包的實現,并進行比較,說明他們的優缺點和應該注意的問題。

Commons FileUpload,可以在http://jakarta.apache.org/commons/fileupload/下載,這個包需要Commons IO的支持,可以在http://jakarta.apache.org/commons/io/下載

com.oreilly.servlet,可以在http://www.servlets.com/cos/下載
Commons FileUpload提供三種文件上傳處理方式,DiskFileUpload、ServletFileUpload和PortletFileUpload三種方式,其中DiskFileUpload已經在javadoc下已經被標記為過期的方法,建議用ServletFileUpload代替,而PortletFileUpload需要配合portlet-api來使用,所以這里我們只介紹ServletFileUpload,并且這個也是最常用的。

com.oreilly.servlet也提供了三種文件上傳的處理方式,MultipartWrapper、MultipartRequest和MultipartParser三種方式,其中MultipartWrapper和MultipartRequest的用法基本相同,并且沒有MultipartRequest提供的操作多,所以這里介紹MultipartRequest,MultipartParser和前兩者有些不同,可以用來處理某些特殊情況,例如表單中有兩個同名的文件上傳選擇框。

我們暫時稱三面三種文件上傳方式分別為:ServletFileUpload方式(MultipartTestServlet)、MultipartRequest方式(MultipartTestServlet2)、MultipartParser方式(MultipartTestServlet3)

代碼如下:
test.html

?

<% @?page?language = " java " ?import = " java.util.* " ?contentType = " text/html;charset=gbk " ?pageEncoding = " gbk " %>
< html >
< body >
< form? action ="MultipartTestServlet" ?enctype ="multipart/form-data" ?method ="post" >
< input? type ="text" ?name ="username" ? />< br? />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="submit" ? />
</ form >
< br />< br />< br />< br />
< form? action ="MultipartTestServlet2" ?enctype ="multipart/form-data" ?method ="post" >
< input? type ="text" ?name ="username" ? />< br? />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="submit" ? />
</ form >
< br />< br />< br />< br />
< form? action ="MultipartTestServlet3" ?enctype ="multipart/form-data" ?method ="post" >
< input? type ="text" ?name ="username" ? />< br? />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="file" ?name ="myfile" ? />< br />
< input? type ="submit" ? />
</ form >
</ body >
</ html >

MultipartTestServlet.java

package ?com.bug.servlet;

import ?java.io.File;
import ?java.io.IOException;
import ?java.util.ArrayList;
import ?java.util.Iterator;
import ?java.util.List;

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.FileUpload;
import ?org.apache.commons.fileupload.FileUploadException;
import ?org.apache.commons.fileupload.RequestContext;
import ?org.apache.commons.fileupload.disk.DiskFileItemFactory;
import ?org.apache.commons.fileupload.servlet.ServletFileUpload;
import ?org.apache.commons.fileupload.servlet.ServletRequestContext;

public ? class ?MultipartTestServlet? extends ?HttpServlet? {

public ?MultipartTestServlet()? {
super ();
}


public ? void ?doPost(HttpServletRequest?request,?HttpServletResponse?response)
throws ?ServletException,?IOException? {

request.setCharacterEncoding(
" gbk " );
RequestContext?requestContext?
= ? new ?ServletRequestContext(request);

if (FileUpload.isMultipartContent(requestContext)) {

DiskFileItemFactory?factory?
= ? new ?DiskFileItemFactory();
factory.setRepository(
new ?File( " c:/tmp/ " ));
ServletFileUpload?upload?
= ? new ?ServletFileUpload(factory);
// upload.setHeaderEncoding("gbk");
upload.setSizeMax( 2000000 );
List?items?
= ? new ?ArrayList();
try ? {
items?
= ?upload.parseRequest(request);
}
? catch ?(FileUploadException?e1)? {
System.out.println(
" 文件上傳發生錯誤 " ? + ?e1.getMessage());
}


Iterator?it?
= ?items.iterator();
while (it.hasNext()) {
FileItem?fileItem?
= ?(FileItem)?it.next();
if (fileItem.isFormField()) {?
System.out.println(fileItem.getFieldName()?
+ ? " ? " ? + ?fileItem.getName()? + ? " ? " ? + ? new ?String(fileItem.getString().getBytes( " iso8859-1 " ),? " gbk " ));
}
else {
System.out.println(fileItem.getFieldName()?
+ ? " ? " ? + ?
fileItem.getName()?
+ ? " ? " ? + ?
fileItem.isInMemory()?
+ ? " ? " ? + ?
fileItem.getContentType()?
+ ? " ? " ? + ?
fileItem.getSize());

if (fileItem.getName() != null ? && ?fileItem.getSize() != 0 ) {
File?fullFile?
= ? new ?File(fileItem.getName());
File?newFile?
= ? new ?File( " c:/temp/ " ? + ?fullFile.getName());
try ? {
fileItem.write(newFile);
}
? catch ?(Exception?e)? {
e.printStackTrace();
}

}
else {
System.out.println(
" 文件沒有選擇?或?文件內容為空 " );
}

}


}

}

}


}


MultipartTestServlet2.java

package ?com.bug.servlet;

import ?java.io.IOException;
import ?java.util.Enumeration;

import ?javax.servlet.ServletException;
import ?javax.servlet.http.HttpServlet;
import ?javax.servlet.http.HttpServletRequest;
import ?javax.servlet.http.HttpServletResponse;

import ?com.oreilly.servlet.MultipartRequest;
import ?com.oreilly.servlet.multipart.DefaultFileRenamePolicy;

public ? class ?MultipartTestServlet2? extends ?HttpServlet? {

public ?MultipartTestServlet2()? {
super ();
}


public ? void ?doPost(HttpServletRequest?request,?HttpServletResponse?response)
throws ?ServletException,?IOException? {

// request.setCharacterEncoding("gbk");?不起作用
System.out.println( " start? " );
MultipartRequest?multi?
= ? new ?MultipartRequest(request,? " c:/tmp/ " ,? 2 * 1024 * 1024 ,? " gbk " ,? new ?DefaultFileRenamePolicy());
System.out.println(
" start? " );
Enumeration?filesName?
= ?multi.getFileNames();
Enumeration?paramsName?
= ?multi.getParameterNames();
while (paramsName.hasMoreElements()) {
String?paramName?
= ?(String)?paramsName.nextElement();
System.out.println(multi.getParameter(paramName));
}

while (filesName.hasMoreElements()) {
String?fileName?
= ?(String)?filesName.nextElement();
System.out.println(multi.getFilesystemName(fileName)?
+ ? " ? " ? +
multi.getOriginalFileName(fileName)?
+ ? " ? " ? + ?
multi.getContentType(fileName)?
+ ? " ? " );
if (multi.getFilesystemName(fileName) != null ? && ? ! multi.getFilesystemName(fileName).equals( "" ))
System.out.println(multi.getFile(fileName).toURI());
}

}


}


MultipartTestServlet3.java

package ?com.bug.servlet;

import ?java.io.File;
import ?java.io.IOException;

import ?javax.servlet.ServletException;
import ?javax.servlet.http.HttpServlet;
import ?javax.servlet.http.HttpServletRequest;
import ?javax.servlet.http.HttpServletResponse;

import ?com.oreilly.servlet.multipart.FilePart;
import ?com.oreilly.servlet.multipart.MultipartParser;
import ?com.oreilly.servlet.multipart.ParamPart;
import ?com.oreilly.servlet.multipart.Part;

public ? class ?MultipartTestServlet3? extends ?HttpServlet? {

public ?MultipartTestServlet3()? {
super ();
}


public ? void ?doPost(HttpServletRequest?request,?HttpServletResponse?response)
throws ?ServletException,?IOException? {

MultipartParser?mp?
= ? new ?MultipartParser(request,? 2 * 1024 * 1024 ,? false ,? false ,? " gbk " );
Part?part;
while ?((part? = ?mp.readNextPart())? != ? null )? {
String?name?
= ?part.getName();
if ?(part.isParam())? {
ParamPart?paramPart?
= ?(ParamPart)?part;
String?value?
= ?paramPart.getStringValue();
System.out.println(
" param:?name= " ? + ?name? + ? " ;?value= " ? + ?value);
}

else ? if ?(part.isFile())? {
// ?it's?a?file?part
FilePart?filePart? = ?(FilePart)?part;
String?fileName?
= ?filePart.getFileName();
if ?(fileName? != ? null )? {
long ?size? = ?filePart.writeTo( new ?File( " c:/tmp/ " ));
System.out.println(
" file:?name= " ? + ?name? + ? " ;?fileName= " ? + ?fileName? +
" ,?filePath= " ? + ?filePart.getFilePath()? + ?
" ,?contentType= " ? + ?filePart.getContentType()? + ?
" ,?size= " ? + ?size);
}

else ? {?
System.out.println(
" file:?name= " ? + ?name? + ? " ;?EMPTY " );
}

System.out.flush();
}

}

}
?

}


web.xml中加入

< servlet >
< servlet-name > MultipartTestServlet </ servlet-name >
< servlet-class > com.bug.servlet.MultipartTestServlet </ servlet-class >
</ servlet >
< servlet >
< servlet-name > MultipartTestServlet2 </ servlet-name >
< servlet-class > com.bug.servlet.MultipartTestServlet2 </ servlet-class >
</ servlet >
< servlet >
< servlet-name > MultipartTestServlet3 </ servlet-name >
< servlet-class > com.bug.servlet.MultipartTestServlet3 </ servlet-class >
</ servlet >
< servlet-mapping >
< servlet-name > MultipartTestServlet </ servlet-name >
< url-pattern > /MultipartTestServlet </ url-pattern >
</ servlet-mapping >
< servlet-mapping >
< servlet-name > MultipartTestServlet2 </ servlet-name >
< url-pattern > /MultipartTestServlet2 </ url-pattern >
</ servlet-mapping >
< servlet-mapping >
< servlet-name > MultipartTestServlet3 </ servlet-name >
< url-pattern > /MultipartTestServlet3 </ url-pattern >
</ servlet-mapping >

問題1、中文問題:
三種凡是都可以通過自己的方法來設置encoding為gbk開處理和解決中文問題,包括初始化的時候傳入gbk作為參數,或是是初始化后通過setEncoding的方式來實現。
另外ServletFileUpload方式也可以通過request.setCharacterEncoding("gbk");的方式來實現,而其它兩種方式不支持這種方式。


問題2、文件大小限制
ServletFileUpload方式可以設置文件大小限制,也可以不用設置,例子中的upload.setSizeMax(2000000)就可以注釋掉。如果設置upload.setSizeMax(-1),表明不限制上傳的大小。文檔中沒有指明默認的限制的多少,我在不設置的情況下上傳了一個9M的東西,可以上傳,估計默認是不限制大小的。
而MultipartRequest方式和MultipartParser方式是必須設置文件的上傳文件的大小限制的,如果不設置,默認是1M的大小限制。


問題3、文件上傳發生錯誤
如果文件上傳過程中發生任何錯誤,或者是文件的大小超出了范圍,系統都將拋出錯誤。
ServletFileUpload方式在upload.parseRequest(request)時拋出錯誤
MultipartRequest方式在new MultipartRequest(。。。)時拋出錯誤
MultipartParser方式在new MultipartParser(。。。)時拋出錯誤


問題4、上傳同名文件時,他們保存到臨時目錄是的沖突問題
ServletFileUpload方式,不會有沖突,系統會把上傳得文件按照一定的規則重新命名,保證不會沖突
MultipartParser方式,會產生沖突,系統會把文件按照上傳時的文件名,保存到臨時目錄下,如果兩個用會同時上傳文件名相同的文件,那么就可能會產生沖突,一方把另一方的臨時文件給替換了。
MultipartRequest方式,在初始化時如果提供了一個名稱轉換策略,就不會有沖突,如果不提桶,就會有沖突。在上面的例子中我們提供了一個new DefaultFileRenamePolicy()保證了文件名不會有沖突發生。


問題5:表單中有兩個同名的文件上傳選擇框,就像我們例子中的myfile一樣,每個表單中有兩個name=“myfile”的上傳框
ServletFileUpload方式,可以處理,可以分別得到他們各自的文件,
MultipartRequest方式,不可以處理,會發生沖突,會有一個上傳框的文件覆蓋了另外一個。
MultipartParser方式,可以處理,可以分別得到他們各自的文件,


備注:
代碼比較亂,主要是為了說明他們間的區別。他們的詳細地使用說明還是要參考他的javadoc和domo。

參考:
1、http://www.servlets.com/cos/#classes
2、http://jakarta.apache.org/commons/fileupload/apidocs/index.html
3、http://jakarta.apache.org/commons/fileupload/using.html
4、http://www.onjava.com/pub/a/onjava/2003/06/25/commons.html?page=3