JavaWeb上傳和下載原理及實現

1、介紹enctype

  • enctype 屬性規定發送到服務器之前應該如何對錶單數據進行編碼。
  • enctype作用是告知服務器請求正文的MIME類型(請求消息頭content-type的作用一樣)

1、1 enctype的取值有三種

描述
application/x-www-form-urlencoded 在發送前編碼所有字符(默認)
multipart/form-data 不對字符編碼。每一個表單項分割爲一個部件
text/plain 空格轉換爲 “+” 加號,但不對特殊字符編碼。

1. 當enctype=’application/x-www-form-urlencoded’

這裏寫圖片描述

2.當enctype=’multipart/form-data’

這裏寫圖片描述
這裏寫圖片描述
通過觀察發現這個的請求體就發生了變化。這種請求體被稱之爲多部件請求體。
什麼是多部件請求體:就是把每一個表單項分割爲一個部件。
以請求頭的content-type的boundary後面的一串隨機字符串作爲分割標識
普通表單項:

//name的意思是文本框裏面name的屬性值,而admin是我們輸入的文本值
Content-Disposition: form-data; name="username"
admin

文件表單項

//filename的意思是:我們上傳的文件名稱,content-Type的意思是:MIME類型,asdasdas的意思是:文件裏面的內容
Content-Disposition: form-data; name="upload"; filename="a.txt"
Content-Type: text/plain
asdasdas

3. 當enctype=’text/plain’

這裏寫圖片描述
w3c稱:空格會變成”+”加號,但是我這裏沒有發現,只有當get請求的時候,空格會變成”+”號
這裏寫圖片描述

進入正題

完成上傳需要滿足3個必要的條件

  1. 提供form表單,method必須是post,因爲get請求的傳輸數據一般爲2kb,不同瀏覽器不一樣。
  2. form表單屬性enctype的必須是multipart/form-data
  3. 提供input type=”file”類的上傳輸入域

    大致實現原理:當enctype的值是multipart/form-data時,瀏覽器會把每個表單項進行分割,分割成不同的部件,以boundary的值爲分割標識,這個標識的字符串是隨機生成的,最後一個表單項的分割標識字符串末尾會多兩個”- -“,代表結束。服務端用request.getHeader(“content-type”)獲取分割字符串,然後進行解析。
    這裏寫圖片描述

代碼實現

一、開發環境搭建

準備兩個第三方jar包
commons-io包
commons-upload包

代碼實現

 protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // 判斷表單的enctype值是不是"multipart/form-data"
        boolean isMultipartContent = ServletFileUpload
                .isMultipartContent(request);
        if (!isMultipartContent) {
            throw new RuntimeException("your form is not multipart/form-data");
        }
        // 創建一個DiskFileItemfactory工廠類
        DiskFileItemFactory factory = new DiskFileItemFactory();
        // 創建一個ServletFileUpload核心對象
        ServletFileUpload sfu = new ServletFileUpload(factory);
        // 解析request對象,並得到一個表單項的集合
        try {
            List<FileItem> fileItems = sfu.parseRequest(request);
            // 遍歷表單項數據
            for (FileItem fileitem : fileItems) {
                if (fileitem.isFormField()) {
                    // 普通表單項
                    String fieldName = fileitem.getFieldName();
                    String fieldValue = fileitem.getString();
                    System.out.println(fieldName + "=====" + fieldValue);
                } else {
                    //上傳表單項
                    //得到文件輸入流
                    InputStream is = fileitem.getInputStream();
                    //創建文件存儲目錄
                    String directoryRealPath = this.getServletContext().getRealPath("WEB-INF/upload");
                    //創建文件路徑
                    File storeDirectory = new File(directoryRealPath + File.separator + fileitem.getName());
                    //使用apache commons-io包,將輸入流轉成文件
                    FileUtils.copyInputStreamToFile(is, storeDirectory);
                }
            }

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

下載的必須條件

兩個頭一個流

content-type

  1. Content-Type是返回消息中非常重要的內容,表示文檔內容屬於什麼MIME類型。
  2. 瀏覽器會根據Content-Type來決定如何顯示返回的消息體內容。
  3. 默認值是text/html
  4. 可以使用request.getServletContext().getMimeType(“文件名”)獲取MIME類型。

Content-Disposition

  1. Content-disposition 是 MIME 協議的擴展,MIME 協議指示 MIME 用戶代理如何顯示附加的文件。
  2. 默認值是inline,表示在瀏覽器窗口中打開。
  3. 服務端向客戶端遊覽器發送文件時,如果是瀏覽器支持的文件類型,一般會默認使用瀏覽器打開,比如txt、jpg等,會直接在瀏覽器 中顯示。
  4. 如果需要提示用戶保存,利用Content-Disposition進行一下處理,關鍵在於一定要加上attachment。
    例如:Content-Disposition:attachment;filename=xxx,瀏覽器就會激活下載框對話框, attachment 表示附件, filname 後面跟隨的是顯示在下載框中的文件名稱。

下載就是向客戶端響應字節數據! 將一個文件變成字節數組, 使用 response.getOutputStream()
來響應給瀏覽器。

實現

    // 下載
    public class DownloadServlet extends HttpServlet{
        public void doGet(HttpServletRequest request, HttpServletResponse response)
                throws ServletException,IOException{
            //根據文件名獲取 MIME 類型
            String contentType = this.getServletContext().getMimeType("測試.xls");

            //設置content-Type頭
            response.setHeader("Content-Type",contentType);

            //解決下載框爲中文名稱出現亂碼問題
            String fileName;

            //獲取用戶瀏覽器
            String agent=request.getHeader("user-agent")

            // 火狐瀏覽器
            if (agent.contains("Firefox")) { 
                filename = "=?UTF-8?B?"
                        + new BASE64Encoder().encode(filename.getBytes("utf-8"))
                        + "?=";
                filename = filename.replaceAll("\r\n", "");
            // IE及其他瀏覽器
            } else { 
                filename = URLEncoder.encode(filename, "utf-8");
                filename = filename.replace("+"," ");
            }
            //設置Content-Disposition頭
            resp.setHeader("Content-Disposition","attachment; filename=" + filename + "");

            // 輸入流
            FileInputStream fis= new FileInputStream("文件路徑");
            // 獲取輸出流
            ServletOutputStream output = response.getOutputStream();
            // 把輸入流中的數據寫入到輸出流中
            byte[] b = new byte[fis.available()];
            fis.read(b);
            //輸出
            output .write(b);
            output .flush();
        }
    }
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章