1. 程式人生 > 其它 >servlet-檔案上傳-狂神筆記

servlet-檔案上傳-狂神筆記

1、面試:檔案上傳調優

  1. 上傳的檔案存放在一個不能使用外界URL訪問的目錄下面
  2. 上傳到同一個資料夾中的檔名稱應該唯一:使用時間戳/UUID/MD5等手段實現
  3. 限制上傳檔案的最大值:因為伺服器上硬碟資源很貴,不能讓使用者隨意的使用
  4. 限制檔案上傳型別:比如這個資料夾只用來儲存圖片,那你就不能上傳一個.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>