1. 程式人生 > 其它 >AcWing1303. 斐波那契前 n 項和(遞推/矩陣快速冪)

AcWing1303. 斐波那契前 n 項和(遞推/矩陣快速冪)

檔案上傳

1、搭建專案

2、導包

檔案在網路上都是使用IO的方式,即流的方式進行的傳輸,而我們要實現的檔案上傳功能可以直接使用apache的元件commons-fileupload(針對檔案上傳的工具類包),這個jar包又依賴commons-io包(封裝了大量的IO操作的工具類),所以在實現檔案上傳功能的時候我們需要匯入這兩個依賴

<dependency>
    <groupId>commons-io</groupId>
    <artifactId>commons-io</artifactId>
    <version>2.9.0</version>
</dependency>
<dependency>
    <groupId>commons-fileupload</groupId>
    <artifactId>commons-fileupload</artifactId>
    <version>1.4</version>
</dependency>

3、使用類介紹

【檔案上傳的注意事項】

  1. .為保證伺服器安全,上傳檔案應該放在外界無法直接訪問的目錄下,比如放於 WEB-INF目錄下。
  2. 為防止檔案覆蓋的現象發生,要為上傳檔案產生一個唯一的檔名:使用時間戳/UUID/MD5等手段實現
  3. 限制上傳檔案的最大值:因為伺服器上硬碟資源很貴,不能讓使用者隨意的使用
  4. 限制檔案上傳型別,在收到上傳檔名時,判斷後綴名是否合法。比如這個資料夾只用來儲存圖片,那你就不能上傳一個.mp4的檔案

檔案上傳在我們生活中很常見,所以一定要掌握

【需要用到的類詳解】

  • ServletFileUpload:在後端獲取檔案上傳的檔案資料,並將上傳檔案資料的表單中的每個輸入項都封裝為一個FileItem物件
  • FileItem
  • DiskFileItemFactory:使用ServletFileUpload解析前端表單傳過來的檔案資料時需要使用到DiskFileItemFactory物件,所以在獲取ServletFileUpload物件之前我們需要先獲取DiskFileItemFactory對
  • fileItemFactory屬性:ServletFileUpload物件的一個屬性

ServletFileUpload負責處理上傳的檔案資料,並將表單中每個輸入項封裝成一個FileItem物件, 在使用ServletFileUpload物件解析請求時需要DiskFileItemFactory物件。所以,我們需要在進行解析工作前構造好DiskFileItemFactory物件,通過ServletFileUpload物件的構造方法或setFileItemFactory()方法設定ServletFileUpload物件的fileItemFactory屬性。

FileItem類

【常用方法介紹】

//isFormField方法用於判斷FileItem類物件封裝的資料是一個普通文字表單
//還是一個檔案表單,如果是普通表單欄位則返回true,否則返回false
boolean isFormField();//即:只要是上傳的檔案,返回的就是false

//getFieldName方法用於返回表單標籤name屬性的值。
String getFieldName();//獲取這個input的name屬性值

//getString方法用於將FileItem物件中儲存的資料流內容以一個字串返回
String getString();//用字串儲存檔案的資料流
    
//getName方法用於獲得檔案上傳欄位中的檔名
String getName();

//以流的形式返回上傳檔案的資料內容。
InputStream getInputStream()

//delete方法用來清空FileItem類物件中存放的主體內容
//如果主體內容被儲存在臨時檔案中,delete方法將刪除該臨時檔案。
void delete();

ServletFileUpload 類

ServletFileUpload負責處理上傳的檔案資料,並將表單中每個輸入項封裝成一個FileItem物件中 . 使用其parseRequest(HttpServletRequest) 方法可以將通過表單中每一個HTML標籤提交的資料封裝成一個FileItem物件,然後以List列表的形式返回,使用該方法處理上傳檔案簡單易用

編寫servlet

檔案上傳JSP

<%@ page contentType="text/html; charset=UTF-8" pageEncoding="UTF-8" %>
<!DOCTYPE html>
<html>
<head>
  <title>JSP - Hello World</title>
</head>
<body>
<h1>檔案上傳</h1>
<br/>
  <%--通過表單上傳檔案--%>
  <%--注意:檔案一般比較大,所以上傳檔案都是使用post方式提交
              get方式只能提交4-5kb的資料,而post沒有限制
              ${pageContext.request.contextPath}:獲取到webapp路徑--%>
  <form action="FileUploadServlet" method="post" enctype="multipart/form-data">
    <p>上傳使用者:<input type="text" name="username"></p>
    <p><input type="file" name="file1"></p>
    <p><input type="file" name="file2"></p>
    <p><input type="submit" value="點選上傳"> | <input type="reset" value="重置"></p>
  </form>
</body>
</html>

顯示檔案上傳成功的跳轉JSP頁面

<%--
  Created by IntelliJ IDEA.
  User: yt
  Date: 2021/7/14
  Time: 14:04
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
    <head>
        <title>Title</title>
    </head>
    <body>
      ${msg}
    </body>
</html>

servlet編寫

package com.yt.servlet; /**
 * ClassName:${NAME}
 * Package:${PACKAGE_NAME}
 * Description:
 *
 * @date:2021/7/14 13:16
 * @author:yt
 */

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;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.annotation.*;
import java.io.*;
import java.util.List;
import java.util.UUID;

@WebServlet(name = "FileUploadServlet", value = "/FileUploadServlet")
public class FileUploadServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request,response);
    }

    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //1、判斷提交的表單是普通表單還是帶上傳檔案的表單
        if(!ServletFileUpload.isMultipartContent(request)){
            //ServletFileUpload.isMultipartContent(req)用於判斷這份表單提交的資料是不是包含檔案
            //結束方法呼叫,說明這是一個普通的表單,沒有包含檔案的<input>,直接返回
            return;
        }

        //2、檔案儲存空間建立:建立上傳檔案的路徑,建議在WEB-INF路徑下,安全,使用者無法直接訪問上傳的檔案
        //獲取/WEB-INF下的/upload路徑
        String uploadPath = this.getServletContext().getRealPath("/WEB-INF/upload");
        //建立一個檔案物件
        File uploadFile = new File(uploadPath);
        if (!uploadFile.exists()){
            //如果這個路徑不存在,就建立這份路徑
            uploadFile.mkdir();
        }

        //3、快取空間建立:臨時路徑,假如檔案超過了預期的大小,我們就把他放在一個臨時檔案中,過幾天自動刪除,或者提醒使用者轉為永久
        String tmpPath = this.getServletContext().getRealPath("WEB-INF/tmp");
        File tmpFile = new File(tmpPath);
        if (!tmpFile.exists()) {
            tmpFile.mkdir();
        }
        DiskFileItemFactory factory = getDiskFileItemFactory(tmpFile);
        ServletFileUpload upload = getServletFileUpload(factory);
        String msg = uploadParasRequest(upload, request, uploadPath);
        System.out.println("===============uploadParasRequest================");

        request.setAttribute("msg",msg);
        request.getRequestDispatcher("info.jsp").forward(request,response);
    }

    public  DiskFileItemFactory getDiskFileItemFactory(File file){
        //直接使用我們所學的流的概念來上傳檔案很低效,且我們的開發成倍很高,所以我們可以直接使用開源的包,比如Apache的檔案上傳元件來實現
        // ,common-fileupload,他需要依賴於commons-io元件

        /*使用規則:
            ①ServletFileUpload負責處理上傳的檔案資料,並將表單中每個輸入項封裝成一個FileItem物件, 在使用ServletFileUpload
        物件解析請求時需要DiskFileItemFactory物件。
            ②我們需要在進行解析工作前構造好DiskFileItemFactory物件,通過ServletFileUpload物件的構造方法或setFileItemFactory()
            方法設定ServletFileUpload物件的fileItemFactory屬性。*/

        //1、建立DiskFileItemFactory物件,獲取磁碟物件
        DiskFileItemFactory factory = new DiskFileItemFactory(1024*1024,file);
        return factory;
    }
    public  ServletFileUpload getServletFileUpload(DiskFileItemFactory factory){
        //2、建立ServletFileUpload物件,獲取上傳檔案的解析物件
        //DiskFileItemFactory物件作為引數傳入ServletFileUpload的構造中
        ServletFileUpload upload = new ServletFileUpload(factory);
        upload.setHeaderEncoding("UTF-8");
        upload.setFileSizeMax(1024*1024*10);
        upload.setSizeMax(1024*1024*10);
        return upload;
    }
    public  String uploadParasRequest(ServletFileUpload upload,HttpServletRequest req,String uploadPath) throws UnsupportedEncodingException {
        String msg = null;
        //3、正式解析表單中上傳的檔案,並將其儲存在伺服器上指定的位置
        try {
            List<FileItem> fileItems = upload.parseRequest(req);
            //使用檔案解析物件的parseRequest()(解析request),這個方法就會將req中的表單項按照一個<input>一個FileItem物件來進行封裝
            //parseRequest(HttpServletRequest) 方法可以將通過表單中每一個HTML標籤提交的資料封裝成一個FileItem物件,然後以List列表的形式返回

            for (FileItem fileItem : fileItems) {//遍歷,找到表單中每一個檔案對應的<input>上傳的檔案資料
                if (fileItem.isFormField()) { //這個<input>中的資料不是檔案
                    String name = fileItem.getFieldName();//獲取非檔案<input>的name屬性
                    String value = fileItem.getString("utf-8");//獲取非檔案<input>的value屬性
                    System.out.println(name + ":" + value);//輸出顯示
                }else { //這個<input>中的資料是檔案

                    //===============1、處理檔案:獲取文上傳的檔案的檔名+檔案型別===============

                    String uploadFileName = fileItem.getName();//獲取這個檔案的名稱
                    System.out.println("上傳的檔名:" + uploadFileName);
                    if (uploadFileName.trim().equals("") || uploadFileName == null) {//如果檔案上傳的名字為空
                        continue;//跳過本次迴圈繼續下一個List元素的遍歷
                    }
                    //精妙點:獲取檔案的名稱
                    //獲得上傳的檔名 /images/girl/paojie.jpg
                    String fileName = uploadFileName.substring(uploadFileName.lastIndexOf("/") + 1);//獲取最後一個/後面的所有字串,獲取結果 = 檔名.檔案型別
                    String fileExtName = uploadFileName.substring(uploadFileName.lastIndexOf(".") + 1);//最後一個"."後面的字串,獲取結果 = 檔案型別
                    System.out.println("檔案資訊[檔名:" + fileName + "----檔案型別" + fileExtName+"]");//列印輸出對檔案進行檢視


                    //=====================2、處理檔案存放地址:/WEB-INF/upload + 檔案上傳時生成的唯一的UUID===============

                    //可以使用UUID(可以唯一識別的通用碼),保證檔名唯一;UUID.randomUUID(),隨機生成一個唯一的識別通用碼;
                    String uuidPath = UUID.randomUUID().toString() ;

                    //存到哪?uploadPath
                    //檔案真正要儲存在伺服器上的存在的路徑realPath = /WEB-INF/upload + 檔案上傳時生成的唯一的UUID
                    String realPath = uploadPath + "/" + uuidPath;
                    //給每個檔案建立一個對應的資料夾
                    File realPathFile = new File(realPath);
                    if (!realPathFile.exists()) {
                        realPathFile.mkdir();   //一般這個檔案儲存的資料夾都是不存在的,所以一定會為我們的上傳檔案建立一個新的資料夾來儲存它
                    }

                    //============================3、檔案傳輸:配合工具類fileName+檔案IO操作就可以實現檔案儲存在伺服器上============================

                    InputStream in = null;//每次遍歷到的都是一個獨立的、完整的檔案對應的fileItem物件,所以我們只需要從它裡面獲取資料流再儲存下來即可
                    try {
                        in = fileItem.getInputStream();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                    //建立一個檔案輸入流
                    FileOutputStream fos = new FileOutputStream(realPath + "/" + fileName);//獲取檔案輸入流
                    //檔名還是和原來保持不變,只是檔案儲存的上一級資料夾的名稱是我們通過/WEB-INF/upload + 檔案上傳時生成的唯一的UUID生成的不重複的
                    byte[] buffer = new byte[1024];//建立一個緩衝區
                    int len = 0;//定義一個變數儲存一次讀到的實際資料量
                    while ((len = in.read(buffer)) > 0) {//通過判斷實際讀取的資料量是不是>0就可以判斷檔案是不是讀完了
                        fos.write(buffer, 0, len);//將檔案流寫到這個檔案中 ——“realPath + "/" + fileName”
                    }
                    //關閉流
                    in.close();
                    fos.close();

                    msg="檔案上傳成功!";
                    fileItem.delete();//上傳成功,清除臨時檔案
                }

            }
        } catch (FileUploadException e) {
            e.printStackTrace();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return msg;
    }
}