婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > JSP實用教程之簡易文件上傳組件的實現方法(附源碼)

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

熱門標簽:外呼系統使用方法 海外圖書館地圖標注點 自繪地圖標注數據 電話機器人需要使用網絡嗎 如何看懂地圖標注點 電銷機器人免培訓 南通通訊外呼系統產品介紹 給地圖標注得傭金 潤滑油銷售電銷機器人

前言

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

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

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

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

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

上傳成功如下圖所示。

使用 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> 

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

源碼下載:http://xiazai.jb51.net/201707/yuanma/SimpleUpload(jb51.net).rar

總結

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

您可能感興趣的文章:
  • jsp頁面驗證碼完整實例
  • JSP實用教程之簡易圖片驗證碼的實現方法(附源碼)
  • JSP中springmvc配置validator的注意事項
  • JSP 開發之Spring Boot 動態創建Bean
  • 詳解SpringBoot集成jsp(附源碼)+遇到的坑
  • 運用JSP+ajax實現分類查詢功能的實例代碼
  • jsp按格式導出doc文件實例詳解

標簽:南京 廣州 大連 貸款邀約 黃石 樂山 銅川 內江

巨人網絡通訊聲明:本文標題《JSP實用教程之簡易文件上傳組件的實現方法(附源碼)》,本文關鍵詞  JSP,實用,教程,之,簡易,文件,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《JSP實用教程之簡易文件上傳組件的實現方法(附源碼)》相關的同類信息!
  • 本頁收集關于JSP實用教程之簡易文件上傳組件的實現方法(附源碼)的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲高清免费一级二级三级| 亚洲精品在线观| 亚洲h精品动漫在线观看| 欧美性色黄大片手机版| 91麻豆精品国产无毒不卡在线观看 | 亚洲欧洲一区二区在线播放| 欧美日韩国产影片| 亚洲电影第三页| 老汉av免费一区二区三区| 久久人人爽人人爽| 欧美性色aⅴ视频一区日韩精品| 精品粉嫩aⅴ一区二区三区四区| 国产精品福利电影一区二区三区四区| 1000部国产精品成人观看| 亚洲美女免费视频| 欧美精品一区二区在线播放| 国产欧美精品区一区二区三区| 欧美吻胸吃奶大尺度电影| 久久伊99综合婷婷久久伊| 日韩一区二区免费在线观看| 欧美激情一区在线观看| 一区二区三区四区不卡在线| 欧美丰满少妇xxxxx高潮对白| 最好看的中文字幕久久| 国产一区不卡在线| 色菇凉天天综合网| 国产精品久久久久久亚洲伦| 国产精品电影一区二区三区| 一区二区三区日韩精品| 欧美精品一区二区三区很污很色的 | 97se亚洲国产综合自在线| 亚洲国产日韩av| 国产精品99久久久久久久女警 | 蜜桃av噜噜一区二区三区小说| 国产精品久久久久久久久免费相片| 国产**成人网毛片九色| 欧美福利视频导航| 91精品国产aⅴ一区二区| 亚洲国产日韩综合久久精品| 日本韩国视频一区二区| 日韩欧美aaaaaa| 亚洲免费毛片网站| 欧美伊人精品成人久久综合97 | 91小视频免费观看| 欧美国产在线观看| 日韩亚洲电影在线| 美女一区二区视频| 成人美女视频在线看| 中日韩免费视频中文字幕| 亚洲精品免费在线| 久久国产精品99久久人人澡| 欧美xxxx老人做受| 亚洲午夜久久久久久久久电影网| 精品国产免费视频| 国产一区二区三区精品欧美日韩一区二区三区 | 国产精品自拍一区| 国产成人亚洲精品青草天美| 91在线小视频| 亚洲永久免费av| 丰满白嫩尤物一区二区| 亚洲美女视频在线观看| 国产综合久久久久久鬼色| 精品国产百合女同互慰| 亚洲男女一区二区三区| 99在线精品免费| 亚洲婷婷在线视频| 日韩精品一区在线| 国产一区二区精品久久99| 国产精品99久| 日韩精品在线一区| 91视频com| 国产精品无码永久免费888| 91年精品国产| 国模娜娜一区二区三区| 中日韩免费视频中文字幕| 91精品国产色综合久久久蜜香臀| 国产成人精品www牛牛影视| 欧美一区二区三区白人| 91色porny在线视频| 亚洲丶国产丶欧美一区二区三区| 国产精品伦理一区二区| 久久蜜桃av一区二区天堂 | 懂色av中文字幕一区二区三区| 日日夜夜免费精品| 亚洲不卡一区二区三区| 1024成人网| 欧美一级xxx| 色婷婷综合视频在线观看| 99久久伊人久久99| 99riav一区二区三区| 欧美色综合网站| 国产精品无人区| 亚洲国产中文字幕在线视频综合| 亚洲成人自拍偷拍| 蜜臀久久久99精品久久久久久| 久久精品久久久精品美女| 国产不卡在线视频| 日本精品一区二区三区高清| 91精品在线一区二区| 欧美大片在线观看一区| 日韩一区二区在线观看视频 | 午夜久久久久久久久| 日韩一区二区三| 欧美a级理论片| 国内成+人亚洲+欧美+综合在线| 一区二区三区欧美久久| 美腿丝袜在线亚洲一区| 欧洲另类一二三四区| 欧美色窝79yyyycom| 99re视频精品| 色噜噜久久综合| 国产成人啪免费观看软件| 麻豆精品国产传媒mv男同| 亚洲第一二三四区| 亚洲精品视频在线| 亚洲国产精品一区二区久久| 午夜精品久久一牛影视| 欧美一级在线视频| 久久精品一区二区三区不卡| 亚洲一区二区综合| 美女在线一区二区| 成人一区在线观看| 91国偷自产一区二区三区成为亚洲经典| 波多野结衣视频一区| 日日摸夜夜添夜夜添国产精品| 一区二区三区中文免费| 国产欧美一区二区精品忘忧草 | 日韩欧美一级精品久久| 欧美一区二区二区| 91久久精品网| 欧美一区中文字幕| 另类小说综合欧美亚洲| 99这里只有精品| 久久久蜜臀国产一区二区| 亚洲欧美色综合| 青青草伊人久久| 99久久免费国产| 欧美一区二区三区白人| 一区二区三区精品久久久| 国产一区二区在线影院| 97se亚洲国产综合自在线不卡 | 欧美高清dvd| 日韩欧美一区二区免费| 精品成人私密视频| 国产精品久久久久精k8| 琪琪一区二区三区| 久久综合九色欧美综合狠狠 | 美女在线视频一区| 一本一道久久a久久精品综合蜜臀| 欧美日韩国产不卡| 一区二区三区在线视频播放| 亚洲一区视频在线观看视频| 亚洲欧美另类久久久精品2019| 麻豆91免费观看| 在线看不卡av| 亚洲综合精品久久| 91浏览器打开| 国产亚洲精品资源在线26u| 欧美日韩一区视频| 亚洲伦理在线免费看| www.亚洲免费av| 国产精品初高中害羞小美女文| 午夜精品久久久久久久99水蜜桃| 成人污视频在线观看| 国产乱子轮精品视频| 欧美群妇大交群的观看方式| 亚洲图片欧美视频| 91在线视频观看| 亚洲午夜私人影院| 国产日产欧美精品一区二区三区| 久久成人久久爱| 久久嫩草精品久久久精品| 成人在线综合网站| 欧美最新大片在线看| 日本韩国欧美一区二区三区| 久久久www成人免费毛片麻豆| 亚洲免费在线播放| 激情伊人五月天久久综合| 精品少妇一区二区三区免费观看| 免费在线观看一区二区三区| 久久综合色鬼综合色| 在线观看国产91| 久久久午夜精品理论片中文字幕| 久久精品99国产精品日本| 久久久久高清精品| 一本一本久久a久久精品综合麻豆 一本一道波多野结衣一区二区 | 日本亚洲一区二区| 欧美日韩成人高清| 久久激情五月婷婷| 亚洲日本韩国一区| 国产精品久久久久天堂| 欧美日韩一区国产| 天天综合网天天综合色| 国产亚洲自拍一区| 国产精品视频在线看| 中文成人av在线| 91精品国产综合久久精品性色| 成人理论电影网| 欧美日韩高清影院| 成人国产精品视频|