servlet-檔案上傳-狂神筆記
阿新 • • 發佈:2021-08-05
1、面試:檔案上傳調優
- 上傳的檔案存放在一個不能使用外界URL訪問的目錄下面
- 上傳到同一個資料夾中的檔名稱應該唯一:使用時間戳/UUID/MD5等手段實現
- 限制上傳檔案的最大值:因為伺服器上硬碟資源很貴,不能讓使用者隨意的使用
- 限制檔案上傳型別:比如這個資料夾只用來儲存圖片,那你就不能上傳一個.mp4的檔案
檔案上傳在我們生活中很常見,所以一定要掌握
2、檔案上傳需要使用到的3個類+1個屬性
- ServletFileUpload:在後端獲取檔案上傳的檔案資料,並將上傳檔案資料的表單中的每個輸入項都封裝為一個FileItem物件
- FileItem
- DiskFileItemFactory:使用ServletFileUpload解析前端表單傳過來的檔案資料時需要使用到DiskFileItemFactory物件,所以在獲取ServletFileUpload物件之前我們需要先獲取DiskFileItemFactory物件
- fileItemFactory屬性:ServletFileUpload物件的一個屬性
ServletFileUpload負責處理上傳的檔案資料,並將表單中每個輸入項封裝成一個FileItem物件, 在使用ServletFileUpload物件解析請求時需要DiskFileItemFactory物件。所以,我們需要在進行解析工作前構造好DiskFileItemFactory物件,通過ServletFileUpload物件的構造方法或setFileItemFactory()方法設定ServletFileUpload物件的fileItemFactory屬性。
3、FileItem類
4、ServletFileUpload 類
ServletFileUpload負責處理上傳的檔案資料,並將表單中每個輸入項封裝成一個FileItem物件中 . 使用其parseRequest(HttpServletRequest) 方法可以將通過表單中每一個HTML標籤提交的資料封裝成一個FileItem物件,然後以List列表的形式返回,使用該方法處理上傳檔案簡單易用
常用方法介紹:
//isFormFile方法用於判斷FileItem類物件封裝的資料是一個普通文字表單 //還是一個檔案表單,如果時普通表單欄位則返回true,否則返回false boolean isField(); //getFieldName方法用於返回表單標籤name屬性的值 String getFieldName(); //getString方法用於將FileItem物件中儲存的資料流內容以一個字串返回 String getString(); //getName方法用於獲得檔案上傳欄位中的檔名。 String getName(); //以流的形式返回上傳檔案的資料內容。 InputStream getInputStream(); //delete方法用來清空FileItem類物件中存放的主體內容 //如果主體內容被儲存在臨時檔案中,delete方法將刪除該臨時檔案 void delete();
程式碼實現:
package com.hao.servlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.UUID;
import org.apache.commons.fileupload.FileItem;
import org.apache.commons.fileupload.FileUploadException;
import org.apache.commons.fileupload.ProgressListener;
import org.apache.commons.fileupload.disk.DiskFileItemFactory;
import org.apache.commons.fileupload.servlet.ServletFileUpload;
public class FileServlet extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws javax.servlet.ServletException, IOException {
//判斷上傳的表單是普通表單還是帶檔案的表單,是返回true,否返回false;
if (!ServletFileUpload.isMultipartContent(request)){
return;//如果這是一個普通檔案我們直接返回
}//如果通過了這個if,說明我們的表單是帶檔案上傳的
//建立上傳檔案的儲存目錄,為了安全建議在WEB-INF目錄下,使用者無法訪問
String uploadpath = this.getServletContext().getRealPath("WEB-INF/Upload");//獲取上傳檔案的儲存路徑
File uploadfile = new File(uploadpath);
if (!uploadfile.exists()){
uploadfile.mkdir();//如果目錄不存在就建立這樣一個目錄
}
//臨時檔案
//臨時路徑,如果上傳的檔案超過預期的大小,我們將它存放到一個臨時目錄中,過幾天自動刪除,或者提醒使用者轉存為永久
String tmppath = this.getServletContext().getRealPath("WEB-INF/tmp");
File file = new File(tmppath);
if (!file.exists()){
file.mkdir();//如果目錄不存在就建立這樣臨時目錄
}
//處理上傳的檔案一般需要通過流來獲取,我們可以通過request.getInputstream(),原生態檔案上傳流獲取,十分麻煩
//但是我們都建議使用Apache的檔案上傳元件來實現,common-fileupload,它需要依賴於common-io元件;
try {
//1、建立DiskFileItemFactory物件,處理檔案上傳路徑或限制檔案大小
DiskFileItemFactory factory = gteDiskFileItemFactory(file);
//2、獲取ServletFileUpload
ServletFileUpload upload = getServletFileUpload(factory);
//3、處理上傳檔案
String msg = uploadParseRequest(upload,request,uploadpath);
//Servlet請求轉發訊息
request.setAttribute("msg",msg);
request.getRequestDispatcher("/info.jsp").forward(request,response);
}catch (FileUploadException e){
e.printStackTrace();
}
}
public static DiskFileItemFactory gteDiskFileItemFactory(File file){
//1、建立DiskFileItemFactory物件,處理檔案上傳路徑或限制檔案大小
DiskFileItemFactory factory = new DiskFileItemFactory();
//通過這個工廠設定一個緩衝區,當上傳的檔案大小大於緩衝區的時候,將它放到臨時檔案中;
factory.setSizeThreshold(1024 * 1024);//緩衝區大小為1M
factory.setRepository(file);
return factory;
}
public static ServletFileUpload getServletFileUpload(DiskFileItemFactory factory){
//2、獲取ServletFileUpload
ServletFileUpload upload = new ServletFileUpload(factory);
//監聽檔案上傳進度
upload.setProgressListener(new ProgressListener() {
public void update(long pBytesRead, long lpContentLenght, int i) {
//pBytesRead:已讀取到的檔案大小
//pContentLenght:檔案大小
System.out.println("總大小:"+lpContentLenght+"已上傳:"+pBytesRead);
}
});
//處理亂碼問題
upload.setHeaderEncoding("UTF-8");
//設定單個檔案的最大值
upload.setFileSizeMax(1024 * 1024 * 10);
//設定總共能夠上傳檔案的大小
//1024 = 1kb * 1024 = 1M * 10 = 10M
upload.setSizeMax(1024 * 1024 * 10);
return upload;
}
public static String uploadParseRequest(ServletFileUpload upload,HttpServletRequest request,String uploadpath) throws IOException, FileUploadException {
String msg = "";
//3、處理上傳檔案
//把前端的請求解析,封裝成一個FileItem物件
List<FileItem> fileItems = upload.parseRequest(request);
for (FileItem fileItem : fileItems) {
if (fileItem.isFormField()){ //判斷是普通表單還是帶檔案的表單
//getFieldName指的是前端表單控制元件的name
String name = fileItem.getFieldName();
String value = fileItem.getString("UTF-8");//處理亂碼
System.out.println(name+":"+value);
}else {//判斷它是帶檔案的表單
//======================處理檔案=======================//
//拿到檔案的名字
String uploadFileName = fileItem.getName();
System.out.println("上傳的檔名:"+uploadFileName);
if (uploadFileName.trim().equals("") || uploadFileName == null){
continue;
}
//獲得上傳的檔名,例如/img/girl/ooa.jpg,只需要ooa,其前面的後面的都不需要
String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);
//獲得檔案的字尾名
String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);
/*
如果檔案字尾名fileExtName不是我們所需要的
就直接return,不處理,告訴使用者檔案型別不對
*/
//可以使用UUID(唯一識別的通用碼),保證檔名唯一
//UUID.randomUUID,隨機生一個唯一識別的通用碼
//網路傳輸中的東西,都需要序列化
//pojo,實體類,如果想要在多個電腦執行,傳輸--->需要吧物件都序列化了
//JNI=java Native Interface
//implements Serializable :標記介面,JVM--->java棧 本地方法棧 native-->c++
String uuidPath= UUID.randomUUID().toString();
System.out.println("檔案資訊【檔名:"+fileName+"檔案型別:"+fileExtName+"】");
//可以使用UUID(唯一通用識別碼)來保證檔名的統一
String uuidFileName = UUID.randomUUID().toString();
//=======================傳輸檔案=========================//
//獲得檔案上傳的流
InputStream inputStream = fileItem.getInputStream();
//建立一個檔案輸出流
FileOutputStream fos = new FileOutputStream(uploadpath + "/" + uuidFileName +"."+ fileExtName);
//建立一個緩衝區
byte[] buffer = new byte[1024 * 1024];
//判斷是否讀取完畢
int len = 0;
//如果大於0,說明還存在資料
while ((len=inputStream.read(buffer))>0){
fos.write(buffer,0,len);
}
//關閉流
fos.close();
inputStream.close();
msg = "檔案上傳成功!";
fileItem.delete();//上傳成功,清除臨時檔案
}
}
return msg;
}
}
jsp頁面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%--通過表單上傳檔案
post:上傳的檔案沒有限制
get:上傳的檔案有限制
--%>
<form action="/upload.do" method="post" enctype="multipart/form-data">
上傳使用者:<input type="text" name="username"> <br>
<p><input type="file" name="file1"></p>
<p><input type="file" name="file2"></p>
<p><input type="submit">|<input type="reset"></p>
</form>
</body>
</html>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
${msg}
</body>
</html>
servlet
<servlet>
<servlet-name>FileServlet</servlet-name>
<servlet-class>com.hao.servlet.FileServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>FileServlet</servlet-name>
<url-pattern>/upload.do</url-pattern>
</servlet-mapping>