亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JSP > 正文

JSP實用教程之簡易文件上傳組件的實現方法(附源碼)

2024-09-05 00:23:14
字體:
來源:轉載
供稿:網友

前言

本文主要給大家介紹的是關于JSP簡易文件上傳組件的實現方法,分享出來供大家參考學習,下面話不多說,來一起看看詳細的介紹吧。

文件上傳,包括但不限于圖片上傳,是 Web 開發中司空見慣的場景,相信各位或多或少都曾寫過這方面相關的代碼。Java 界若說文件上傳,則言必稱 Apache Commons FileUpload,論必及  SmartUpload。更甚者,Servlet 3.0 將文件上傳列入 JSR 標準,使得通過幾個注解就可以在 Servlet 中配置上傳,無須依賴任何組件。使用第三方組件或 Servlet 自帶組件固然強大,但只靠 JSP 亦能完成任務,且短小而精悍,豈不美哉?本文實現的方法純然基于 JSP 代碼,沒有弄成 Servlet 和專門的 Class(.java),實現方法純粹是基于 JSP,沒有太高的技術難度。實際使用過程中直接部署即可。

操作組件的代碼行數不超過 10 行,只需幾個步驟:

  • 生成組件實例
  • 設置實例屬性
  • 調用上傳/下載方法
  • 處理調用結果

首先是上傳頁面,本例是一張靜態的 HTML。

jsp,文件上傳,jsp文件上傳代碼,jsp文件上傳下載組件

上傳成功如下圖所示。

jsp,文件上傳,jsp文件上傳代碼,jsp文件上傳下載組件

使用 POST 的表單,設置 ContentType 為 multipart/form-data 多段數據,還要記得 input 的 name 屬性。

<html> <body>  <form action="action.jsp" enctype="multipart/form-data" method="POST">   selectimage: <input type="file" name="myfile" /><br> <input    type="submit" value="upload" />  </form> </body> </html> 

action 中接受客戶端請求的服務端代碼在 action.jsp 中。action.jsp 通過 <%@include file="Upload.jsp"%>包含了核心 Java 代碼,而 Upload.jsp 里面又包含了另外一個 UploadRequest.jsp 文件??傊?,我們這個小小的 Java 程序,一共包含了 UploadRequest 請求信息類、UploadException 自定義異常類和最重要的 Upload 類這三個類。

<%@page pageEncoding="UTF-8"%> <%@include file="Upload.jsp"%> <%  UploadRequest ur = new UploadRequest();// 創建請求信息,所有參數都在這兒設置  ur.setRequest(request); //一定要傳入 request  ur.setFileOverwrite(true);// 相同文件名是否覆蓋?true=允許覆蓋   Upload upload = new Upload();// 上傳器   try {   upload.upload(ur);  } catch (UploadException e) {   response.getWriter().println(e.toString());  }   if (ur.isOk()) // 上傳成功   response.getWriter().println("上傳成功:" + ur.getUploaded_save_fileName());  else   response.getWriter().println("上傳失??!"); %> 

這里創建了 UploadRequest 實例。文件上傳操作通常會附加一些限制,如:文件類型、上傳文件總大小、每個文件的最大大小等。除此以外,作為一個通用組件還需要考慮更多的問題, 如:支持自定義文件保存目錄、支持相對路徑和絕對路徑、支持自定義保存的文件的文件名稱等。這些配置通通在 UploadRequest 里設置。

至于 JSP 里面的類,我愿意多說說。 JSP 里面允許我們定義 Java 的類,類本是可以是 static,但不能有 static 成員。實際上 JSP 類都是內部類,定義 static 與否關系不大。如果不能定義 static 方法,就把 static 方法移出類體外,書寫成,

<%!   /**  * 獲取開頭數據頭占用的長度  *  * @param dateBytes  *   文件二進制數據  * @return  */  private static int getStartPos(byte[] dateBytes) {    ....   }  > 

<%! ... %><% ... %> 不同,前者是定義類成員的。

好~我們在看看 UploadRequest.jsp,就知道具體配置些什么。

<%@page pageEncoding="UTF-8"%> <%!/**   * 上傳請求的 bean,包含所有有關請求的信息   * @author frank   *   */  public static class UploadRequest {   /**    * 上傳最大文件大小,默認 1 MB    */   private int MaxFileSize = 1024 * 1000;    /**    * 保存文件的目錄    */   private String upload_save_folder = "E://temp//";    /**    * 上傳是否成功    */   private boolean isOk;    /**    * 是否更名    */   private boolean isNewName;    /**    * 成功上傳之后的文件名。如果 isNewName = false,則是原上傳的名字    */   private String uploaded_save_fileName;    /**    * 相同文件名是否覆蓋?true=允許覆蓋    */   private boolean isFileOverwrite = true;    private HttpServletRequest request;    /**    * @return the maxFileSize    */   public int getMaxFileSize() {    return MaxFileSize;   }    /**    * @param maxFileSize the maxFileSize to set    */   public void setMaxFileSize(int maxFileSize) {    MaxFileSize = maxFileSize;   }    /**    * @return the upload_save_folder    */   public String getUpload_save_folder() {    return upload_save_folder;   }    /**    * @param upload_save_folder the upload_save_folder to set    */   public void setUpload_save_folder(String upload_save_folder) {    this.upload_save_folder = upload_save_folder;   }    /**    * @return the isOk    */   public boolean isOk() {    return isOk;   }    /**    * @param isOk the isOk to set    */   public void setOk(boolean isOk) {    this.isOk = isOk;   }    /**    * @return the isNewName    */   public boolean isNewName() {    return isNewName;   }    /**    * @param isNewName the isNewName to set    */   public void setNewName(boolean isNewName) {    this.isNewName = isNewName;   }    /**    * @return the uploaded_save_fileName    */   public String getUploaded_save_fileName() {    return uploaded_save_fileName;   }    /**    * @param uploaded_save_fileName the uploaded_save_fileName to set    */   public void setUploaded_save_fileName(String uploaded_save_fileName) {    this.uploaded_save_fileName = uploaded_save_fileName;   }    /**    * @return the isFileOverwrite    */   public boolean isFileOverwrite() {    return isFileOverwrite;   }    /**    * 相同文件名是否覆蓋?true=允許覆蓋    * @param isFileOverwrite the isFileOverwrite to set    */   public void setFileOverwrite(boolean isFileOverwrite) {    this.isFileOverwrite = isFileOverwrite;   }    /**    * @return the request    */   public HttpServletRequest getRequest() {    return request;   }    /**    * @param request the request to set    */   public void setRequest(HttpServletRequest request) {    this.request = request;   }   }   %> 

這是一個普通的 java bean。完成上傳邏輯的是 Upload 類。

其原理是:

1、由客戶端把要上傳的文件生成 request 數據流,與服務器端建立連接;

2、在服務器端接收 request 流,將流緩存到內存中;

3、由服務器端的內存把文件輸出到指定的目錄。

Upload.jsp 完整代碼如下所示。

<%@page pageEncoding="UTF-8" import="java.io.*"%> <%@include file="UploadRequest.jsp"%> <%!  public static class UploadException extends Exception {    private static final long serialVersionUID = 579958777177500819L;   public UploadException(String msg) {   super(msg);  }  }  public static class Upload {  /**   * 接受上傳   *   * @param uRequest   *   上傳 POJO   * @return   * @throws UploadException   */  public UploadRequest upload(UploadRequest uRequest) throws UploadException {   HttpServletRequest req = uRequest.getRequest();      // 取得客戶端上傳的數據類型   String contentType = req.getContentType();    if(!req.getMethod().equals("POST")){    throw new UploadException("必須 POST 請求");   }      if (contentType.indexOf("multipart/form-data") == -1) {    throw new UploadException("未設置表單 multipart/form-data");   }      int formDataLength = req.getContentLength();      if (formDataLength > uRequest.getMaxFileSize()) { // 是否超大    throw new UploadException("文件大小超過系統限制!");   }      // 保存上傳的文件數據   byte dateBytes[] = new byte[formDataLength];   int byteRead = 0, totalRead = 0;    try(DataInputStream in = new DataInputStream(req.getInputStream());){    while (totalRead < formDataLength) {     byteRead = in.read(dateBytes, totalRead, formDataLength);     totalRead += byteRead;    }   } catch (IOException e) {    e.printStackTrace();    throw new UploadException(e.toString());   }            // 取得數據分割字符串   int lastIndex = contentType.lastIndexOf("="); // 數據分割線開始位置boundary=---------------------------   String boundary = contentType.substring(lastIndex + 1, contentType.length());// ---------------------------257261863525035    // 計算開頭數據頭占用的長度   int startPos = getStartPos(dateBytes);   // 邊界位置   int endPos = byteIndexOf(dateBytes, boundary.getBytes(), (dateBytes.length - startPos)) - 4;    // 創建文件   String fileName = uRequest.getUpload_save_folder() + getFileName(dateBytes, uRequest.isNewName());   uRequest.setUploaded_save_fileName(fileName);   File checkedFile = initFile(uRequest);    // 寫入文件   try(FileOutputStream fileOut = new FileOutputStream(checkedFile);){    fileOut.write(dateBytes, startPos, endPos - startPos);    fileOut.flush();        uRequest.setOk(true);   } catch (FileNotFoundException e) {    e.printStackTrace();    throw new UploadException(e.toString());   } catch (IOException e) {    e.printStackTrace();    throw new UploadException(e.toString());   }      return uRequest;  } }   /**   * 獲取開頭數據頭占用的長度   *   * @param dateBytes   *   文件二進制數據   * @return   */  private static int getStartPos(byte[] dateBytes) {   int startPos;   startPos = byteIndexOf(dateBytes, "filename=/"".getBytes(), 0);   startPos = byteIndexOf(dateBytes, "/n".getBytes(), startPos) + 1; // 遍歷掉3個換行符到數據塊   startPos = byteIndexOf(dateBytes, "/n".getBytes(), startPos) + 1;   startPos = byteIndexOf(dateBytes, "/n".getBytes(), startPos) + 1;      return startPos;  }    /**   * 在字節數組里查找某個字節數組,找到返回>=0,未找到返回-1   * @param data   * @param search   * @param start   * @return   */  private static int byteIndexOf(byte[] data, byte[] search, int start) {   int index = -1;   int len = search.length;   for (int i = start, j = 0; i < data.length; i++) {    int temp = i;    j = 0;    while (data[temp] == search[j]) {     // System.out.println((j+1)+",值:"+data[temp]+","+search[j]);     // 計數     j++;     temp++;     if (j == len) {      index = i;      return index;     }    }   }   return index;  }    /**   * 如果沒有指定目錄則創建;檢測是否可以覆蓋文件   *   * @param uRequest   *   上傳 POJO   * @return   * @throws UploadException   */  private static File initFile(UploadRequest uRequest) throws UploadException {   File dir = new File(uRequest.getUpload_save_folder());   if (!dir.exists())    dir.mkdirs();      File checkFile = new File(uRequest.getUploaded_save_fileName());      if (!uRequest.isFileOverwrite() && checkFile.exists()) {    throw new UploadException("文件已經存在,禁止覆蓋!");   }      return checkFile;  }    /**   * 獲取 POST Body 中的文件名   *   * @param dateBytes   *   文件二進制數據   * @param isAutoName   *   是否自定命名,true = 時間戳文件名   * @return   */  private static String getFileName(byte[] dateBytes, boolean isAutoName) {   String saveFile = null;      if(isAutoName){    saveFile = "2016" + System.currentTimeMillis();   } else {    String data = null;    try {     data = new String(dateBytes, "UTF-8");    } catch (UnsupportedEncodingException e) {     e.printStackTrace();     data = "errFileName";    }        // 取得上傳的文件名    saveFile = data.substring(data.indexOf("filename=/"") + 10);    saveFile = saveFile.substring(0, saveFile.indexOf("/n"));    saveFile = saveFile.substring(saveFile.lastIndexOf("//") + 1, saveFile.indexOf("/""));   }      return saveFile;  } %> 

通過 DataInputStream 讀取流數據到 dataBytes 中然后寫入 FileOutputStream。另外還有些圍繞配置的邏輯。

值得一提的是,Tomcat 7 下 JSP 默認的 Java 語法仍舊是 1.6 的。在 JSP 里面嵌入 Java 1.7 特性的代碼會拋出“Resource specification not allowed here for source level below 1.7”的異常。于是需要修改 Tomcat/conf/web.xml 里面的配置文件,找到 <servlet> 節點,加入下面粗體部分才可以。注意是 jsp 節點,不是 default 節點(很相似)。

<servlet>   <servlet-name>jsp</servlet-name>   <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class>   <init-param>    <param-name>fork</param-name>    <param-value>false</param-value>   </init-param>   <init-param>    <param-name>xpoweredBy</param-name>    <param-value>false</param-value>   </init-param> lt;strong>  <init-param>    <param-name>compilerSourceVM</param-name>    <param-value>1.7</param-value>   </init-param>   <init-param>    <param-name>compilerTargetVM</param-name>    <param-value>1.7</param-value>   </init-param></strong>   <load-on-startup>3</load-on-startup>  </servlet> 

至此,一個簡單的文件上傳器就完成了。但是本組件的缺點還是很明顯的,試列舉兩項:一、上傳流占用內存而非磁盤,所以上傳大文件時內存會吃緊;二、尚不支持多段文件上傳,也就是一次只能上傳一個文件。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網的支持。


注:相關教程知識閱讀請移步到JSP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲免费精彩视频| 亚洲人成欧美中文字幕| 97国产精品免费视频| 国产欧美日韩中文字幕| 久久久91精品国产一区不卡| 亚洲欧洲美洲在线综合| 久久久免费电影| 久久九九有精品国产23| 久久黄色av网站| 成人a视频在线观看| 国产视频久久久| 日本亚洲欧洲色α| 欧美色videos| 最新国产精品亚洲| 日韩www在线| 成人免费网视频| 精品国产一区二区三区久久| 国产精品欧美一区二区| 人九九综合九九宗合| 精品高清美女精品国产区| 欧美国产精品va在线观看| 日韩欧美高清在线视频| 91精品视频在线播放| 午夜欧美不卡精品aaaaa| 欧美大片在线免费观看| 69久久夜色精品国产69乱青草| 夜色77av精品影院| 国产精品美女999| 国产91成人video| 亚洲18私人小影院| 国产精品视频网址| 日韩第一页在线| 亚洲精品福利视频| 久久综合电影一区| 日韩一区在线视频| 青青在线视频一区二区三区| 国产精品久久色| 欧美性生交xxxxxdddd| 性视频1819p久久| 亚洲精品欧美日韩| 国产伦精品一区二区三区精品视频| 久久99国产综合精品女同| 青草青草久热精品视频在线网站| 欧美日韩精品在线| 色综合色综合久久综合频道88| 欧美激情精品久久久久久黑人| 欧美精品一区二区三区国产精品| 精品国产乱码久久久久酒店| 日韩欧美大尺度| 亚洲欧美日韩一区二区三区在线| 国产亚洲精品91在线| 国产精品久久久久久久久免费看| 国产aaa精品| 国产精品成人观看视频国产奇米| 亚洲国产古装精品网站| 欧洲日本亚洲国产区| 亚洲国产另类 国产精品国产免费| 一区二区三区www| 国产日韩欧美在线看| 久久久精品在线| 日韩美女激情视频| 精品国内亚洲在观看18黄| 精品一区二区亚洲| 国产精品欧美日韩| 亚洲热线99精品视频| 国模精品视频一区二区| 亚洲新声在线观看| 久久久久久久亚洲精品| 日韩国产欧美精品在线| 久久精品小视频| 亚洲综合av影视| 欧美一级淫片播放口| 亚洲欧美日韩中文在线制服| 午夜精品一区二区三区在线视| 国产黑人绿帽在线第一区| 亚洲v日韩v综合v精品v| 91免费版网站入口| 亚洲天堂av高清| 91av在线精品| 777国产偷窥盗摄精品视频| 欧美激情视频免费观看| 精品美女久久久久久免费| 亚洲第一天堂无码专区| 狠狠躁天天躁日日躁欧美| 亚洲欧美在线x视频| 国内精品伊人久久| 久久国产精品久久精品| 国产97在线观看| 久久精品中文字幕电影| 97久久精品人人澡人人爽缅北| 欧美成年人视频网站| 九九视频这里只有精品| 中文字幕一区二区三区电影| 美女久久久久久久| 欧洲美女免费图片一区| 欧美网站在线观看| 68精品国产免费久久久久久婷婷| 中文字幕免费精品一区| 国产福利精品在线| 欧洲精品在线视频| 亚洲国产精品999| 国产精品久久久久77777| 日本精品免费观看| 亚洲视频国产视频| 国产精品久久一区主播| 懂色av一区二区三区| 亚洲欧美日韩天堂| 欧美乱大交做爰xxxⅹ性3| 国产91精品视频在线观看| 亚洲精品欧美极品| 精品久久久久久久久久久久| 欧美电影在线观看完整版| 性欧美视频videos6一9| 欧美一级视频一区二区| 欧美精品手机在线| 成人免费午夜电影| 亚洲精品自拍第一页| 日韩在线一区二区三区免费视频| 亚洲人成在线观看网站高清| 欧美国产中文字幕| 最近日韩中文字幕中文| 91精品国产91久久久久久吃药| 欧美大尺度激情区在线播放| 亚洲电影免费观看高清完整版在线观看| 中文字幕亚洲无线码a| 国产小视频91| 欧美激情视频网站| 欧美国产亚洲精品久久久8v| 视频直播国产精品| 亚洲小视频在线观看| 最新亚洲国产精品| 精品视频偷偷看在线观看| 国外色69视频在线观看| 久久久久日韩精品久久久男男| 国产v综合v亚洲欧美久久| 亚洲精选中文字幕| 亚洲综合中文字幕68页| 亚洲精品wwwww| 亚洲欧美日韩爽爽影院| 久久国产精品亚洲| 久久久久久久久久久av| 亚洲已满18点击进入在线看片| 亚洲精品99999| 一本久久综合亚洲鲁鲁| 一区二区福利视频| 国产精品99久久久久久人| 日韩精品免费综合视频在线播放| 亚洲国产精品久久91精品| 成人高清视频观看www| 日本国产高清不卡| 国产伦精品一区二区三区精品视频| 国产成人久久久精品一区| 欧美中文在线观看国产| 亚洲一区制服诱惑| 日韩视频在线免费| 欧美自拍视频在线| 中文欧美日本在线资源| 亚洲日本成人女熟在线观看| 国产精品2018| 日韩高清中文字幕| 欧美精品18videosex性欧美| 成人97在线观看视频| 亚洲成av人影院在线观看| 性欧美亚洲xxxx乳在线观看|