Web裏面的文件上傳

這裏我們來總結整理一下常用的兩種文件上傳方式以及要注意的東西:

1、springmvc .MultipartFile 的上傳方式。

2、org.apache.commons.fileupload 使用apache的fileuoload 來實現

當我們使用springmvc 的MultipartFile 時我們要在spring的配置文件中添加如下的配置文件:

(1)、bean的引入以及編碼和上傳文件大小兩個屬性的設置

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. <bean id="multipartResolver"    
  2.         class="org.springframework.web.multipart.commons.CommonsMultipartResolver">    
  3.         <property name="defaultEncoding" value="UTF-8" />    
  4.     
  5.         <property name="maxUploadSize" value="2000000000" />    
  6.     </bean>    

(2)、控制層的代碼:

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. public void upload2(HttpServletRequest request) {    
  2.         // 轉型爲MultipartHttpRequest    
  3.         try {    
  4.             MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;    
  5.             List<MultipartFile> fileList = multipartRequest.getFiles("file");    
  6.             for (MultipartFile mf : fileList) {    
  7.                 if(!mf.isEmpty()){    
  8.                         
  9.                 }    
  10.             }    
  11.         } catch (Exception e) {    
  12.             e.printStackTrace();    
  13.         }    
  14.             
  15.     }    

這樣就簡單的獲取了上傳文件,其實我們上傳的文件就是一個MultipartFile 對象,獲取了對象後我們就可以獲取文件名以及輸入流。

(3)、控制層的第二種方式:

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. public String upload(HttpServletRequest request,    
  2.             @RequestParam(value = "file") MultipartFile[] files) {    
  3.         try {    
  4.             for (MultipartFile mf : files) {    
  5.                 if(!mf.isEmpty()){    
  6.                         
  7.                 }    
  8.             }    
  9.     
  10.         } catch (Exception e) {    
  11.             e.printStackTrace();    
  12.         }    
  13.         return "upload";    
  14.     }    

(2)、(3)、不管哪種,其實都是一樣的原理,都是將上傳文件轉化爲了對象。

這裏springMVC 都爲我們封裝好成自己的文件對象了,轉換的過程就在我們所配置的CommonsMultipartResolver這個轉換器裏面下面再來看看它的源碼


看到這個你應該明白了,如果你配置了multipartResolver 這個bean 則程序會自動調用fileUpload 來解析request ,在控制層獲取的request已經是被解析過的,所以不用在手動去調用fileUpload 的相關方法來解析這個request。但注意,如果在spring的配置文件中沒有配置這個bean的話,則我們自己要在controller層調用fileUpload 的相關方法來解析request 。ok這裏說了第一種使用springmvc 的MultipartFile ,但是第二種和第一種其實是一樣的,第一種解析的過程由spring替我們幹了,我們只需要獲取數據就ok而第二種方式就是在spring 的配置文件中不配置bean 這樣在控制層中我們只能自己去解析請求。

所以簡單的來說其實就是一種技術的兩種使用方式,第一種是配置在了spring 容器中而第二種就是不配置在spring容器中而是自己去解析請求信息。


注意:剛開始學習的時候,你肯能不瞭解而出現了獲取不到數據的這種情況,那就是在你不知不覺中解析了兩次request所以你獲取不到你想要的數據,那怎麼處理那其實很簡單,具體看下面:

1、不在spring的配置文件中配置,則需要你在控制層調用fileupload 的方法進行解析:

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. @RequestMapping(value="/upload.do")  
  2.     @SuppressWarnings("unchecked")  
  3.     public void upload(HttpServletRequest  request,HttpServletResponse response) throws IOException, FileUploadException{  
  4.          System.out.println("文件存放目錄、臨時文件目錄準備完畢 ...");    
  5.          System.out.println("filePath-->:"+filePath);  
  6.          System.out.println("tempPath-->:"+tempPath);  
  7.          response.setContentType("text/plain;charset=utf-8");    
  8.          PrintWriter pw = response.getWriter();  
  9.             // 磁盤文件工廠,創建緩存文件  
  10.              DiskFileItemFactory diskFactory = new DiskFileItemFactory();    
  11.              // threshold 極限、臨界值,即磁盤緩存大小設置  
  12.              diskFactory.setSizeThreshold(10*4 * 1024);    
  13.              // repository 貯藏室,即臨時文件目錄 ,設置文件的緩存路徑  
  14.              diskFactory.setRepository(new File(tempPath));    
  15.              ServletFileUpload upload = new ServletFileUpload(diskFactory);    
  16.              // 設置允許上傳的最大文件大小 ,如果是-1表示沒有限制  
  17.              upload.setSizeMax(10 * 1024 * 1024);    
  18.              //解決上傳的文件名亂碼  
  19.              upload.setHeaderEncoding("UTF-8");  
  20.              // 3、判斷用戶的表單提交方式是不是multipart/form-data,這也是我們在form中設置的那個屬性的值 是不是……  
  21.              boolean bb = upload.isMultipartContent(request);  
  22.               if (!bb) {  
  23.                      return ;  
  24.                  }  
  25.              // 解析HTTP請求消息頭 ,也就是調用方法解析提交的內容並將其組裝成一個個的FileItem對象  
  26.              // 而其中上傳的一個文件就是一個FileItem對象  
  27.              List<FileItem> fileItems = upload.parseRequest(request);  
  28.              try {  
  29.                 Iterator<FileItem> iter = fileItems.iterator();    
  30.                   while(iter.hasNext())    
  31.                     {     
  32.                         FileItem item = (FileItem)iter.next();    
  33.                          //按照給定的編碼格式獲取上傳文件內容  
  34.                         String fieldValue = item.getString("UTF-8");  
  35.                         //獲取標籤名稱  
  36.                         String tagName = item.getFieldName();  
  37.                         //獲取文件名稱  
  38.                         String fileName= item.getName();  
  39.                         //上傳文件輸入流,也就是整個上傳文件的流  
  40.                         InputStream  input =  item.getInputStream();  
  41.                         System.out.println("tagName--->:"+tagName   +"fileName--->"+fileName );  
  42.                         //判斷這個FileItem是不是表單屬性(他能判斷是上傳的文件還是表單屬性)  
  43.                         if(item.isFormField())    
  44.                         {    
  45.                             System.out.println("處理表單內容 ...");    
  46.                             processFormField(item, pw);    
  47.                         }else{    
  48.                             System.out.println("處理上傳的文件 ...");    
  49.                             processUploadFile(item, pw);    
  50.                         }    
  51.                     }  
  52.             } catch (Exception e) {  
  53.                 e.printStackTrace();  
  54.             }finally{  
  55.                 pw.close();  
  56.             }    
  57.           
  58.     }  

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. /** 
  2.   * 處理表單內容   
  3.   * @param item 
  4.   * @param pw 
  5.   * @throws Exception 
  6.   */  
  7.   private void processFormField(FileItem item, PrintWriter pw)    
  8.       throws Exception    
  9.   {    
  10.       String tagName = item.getFieldName();    
  11.       String fileName = item.getString("utf-8");            
  12.       pw.println(tagName + " : " + fileName + "\r\n");    
  13.   }    

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1.  /** 
  2.     * 處理上傳的文件   
  3.     * @param item 
  4.     * @param pw 
  5.     * @throws Exception 
  6.     */  
  7. private void processUploadFile(FileItem item, PrintWriter pw)    
  8.        throws Exception    
  9.    {    
  10.        //文件名稱  
  11.        String filename = item.getName();           
  12.        System.out.println("完整的文件名:" + filename);    
  13.        //上傳文件大小(byte)  
  14.        long fileSize = item.getSize();    
  15.        if("".equals(filename) && fileSize == 0)    
  16.        {               
  17.            System.out.println("文件名爲空 ...");    
  18.            return;    
  19.        }    
  20.          System.out.println("filePath:-->"+filePath);  
  21.          System.out.println("filename:--->"+filename);  
  22.          //創建保存文件路徑  
  23.        File uploadFile = new File(filePath +File.separator + filename);    
  24.        if(!uploadFile.getParentFile().exists()){  
  25.         uploadFile.getParentFile().mkdirs();  
  26.        }  
  27.        uploadFile.createNewFile();  
  28.        //將上傳上來的文件內容寫到指定的文件  
  29.        item.write(uploadFile);    
  30.        //向瀏覽器打印  
  31.        pw.println(filename + " 文件保存完畢 ...");    
  32.        pw.println("文件大小爲 :" + fileSize + "\r\n");    
  33.    }   

2、在spring的配置文件中配置了這個bean 則不在需要我們自己調用一個fileupload 的方法再去解析一次,而是直接使用:

[java] view plain copy
 print?在CODE上查看代碼片派生到我的代碼片
  1. /** 
  2.      * spring mvc MultipartFile 多文件上傳 
  3.      * @param request 
  4.      * @param response 
  5.      * @throws IOException 
  6.      */  
  7.     @RequestMapping(value="/upload1.do")  
  8.     public void  upload1(HttpServletRequest  request,HttpServletResponse response) throws IOException{  
  9.          //創建一個通用的多部分解析器    
  10.         CommonsMultipartResolver multipartResolver = new CommonsMultipartResolver(request.getSession().getServletContext());    
  11.         //判斷用戶的表單提交方式是不是multipart/form-data,這也是我們在form中設置的那個屬性的值 是不是……  
  12.         if(multipartResolver.isMultipart(request)){    
  13.         //判斷 request 是否有文件上傳,即多部分請求    
  14.          MultipartHttpServletRequest multipartRequest = (MultipartHttpServletRequest) request;   
  15.          //獲取所有上傳文件的名稱是input標籤中的名稱  
  16.          Iterator<String> iter = multipartRequest.getFileNames();   
  17.          while(iter.hasNext()){  
  18.              String name = iter.next();  
  19.              //按照文件名稱獲取這個上傳文件,上傳文件被轉化爲MultipartFile 對象  
  20.              MultipartFile  file= multipartRequest.getFile(name);  
  21.              //獲取文件名,這裏的fname也是input標籤中的name而不是文件名  
  22.              String fname= file.getName();  
  23.              //這裏獲取到的纔是真真的文件名稱 比如test.txt等這樣的名稱。  
  24.               String myfname= file.getOriginalFilename();  
  25.              //獲取輸入流  
  26.              InputStream  input = file.getInputStream();  
  27.             System.out.println("fname: "+fname  +" name: "+name +" myfname:"+myfname);  
  28.          }  
  29.           //這的getFiles("file")就是獲取多個input標籤名稱爲file的文件。  
  30.           List<MultipartFile> fileList = multipartRequest.getFiles("file");    
  31.          for(MultipartFile file:fileList){  
  32.             System.out.println("----fileList size:"+fileList.size());  
  33.             //獲取input標籤中的名稱  
  34.             String name= file.getName();  
  35.             //這裏獲取到的纔是真真的文件名稱 比如test.txt等這樣的名稱。  
  36.             String  myfname= file.getOriginalFilename();  
  37.             //獲取這個文件的輸入流  
  38.              InputStream  input = file.getInputStream();  
  39.             System.out.println("name"+name +"myfname:"+myfname);  
  40.          }  
  41.         }  
  42.         PrintWriter p= response.getWriter();  
  43.         p.write("ok!!!");  
  44.     }  
發佈了24 篇原創文章 · 獲贊 3 · 訪問量 6萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章