文件或資源的上傳和下載
1.文件的上傳
[1] 簡介
> 將一個客戶端的本地的文件發送到服務器中保存。
> 上傳文件是通過流的形式將文件發送給服務器。
[2] 表單的設置
> 向服務器上傳一個文件時,表單要使用post請求。
> 表單的默認屬性enctype="application/x-www-form-urlencoded"
- 這個屬性的意思是請求體中的內容將會使用URL編碼
> 上傳文件的表單enctype需要設置為 multipart/form-data
- multipart/form-data表示的是表單是一個多部件的表單
- 如果類型設置為它,則我們的每一個表單項都會作為一個單獨的部件發送給服務器。
- 多個部件之間使用類似 -----------------------------7df2d08c0892 分割符來分開
> 當表單設置為multipart/form-data時,我們request.getParameter()將失效,我們不能再通過該方法獲取請求參數。
[3] FileUpload
> 我們一般情況下使用commons-fileupload-1.3.1.jar這個工具來解析多部件請求。
> fileupload 依賴 commons-io 所以我們要是Filtupload需要同時導入io包。
> 核心類:
DiskFileItemFactory
- 工廠類,用於構建一個解析器實例。
ServletFileUpload
- 解析器類,通過該類實例來解析request中的請求信息。
FileItem
- 工具會將我們請求中每一個部件,都封裝為一個FileItem對象,處理文件上傳時,只需要調用該對象的方法
- 方法:
boolean isFormField() --> 當前表單項是否是一個普通表單項,true是普通表單項, false是文件表單項
String getContentType() --> 返回的是文件的類型,是MIME值
String getFieldName() --> 獲取表單項的name屬性值
String getName() --> 獲取上傳的文件的名字
long getSize() --> 獲取文件的大小
String getString(String encoding) --> 獲取表單項的value屬性值,需要接受一個編碼作為參數。
void write(File file) --> 將表單項中的內容寫入到磁盤中
> 使用步驟:
1.獲取工廠類實例[DiskFileItemFactory]
2.獲取解析器類實例[ServletFileUpload]
3.解析request獲取FileItem[parseRequest()]
[4] 細節
第一個問題
> 部分瀏覽器會將文件的完整路徑作為文件名發送。
C:\Users\lilichao\Desktop\day20\圖片\蒙娜麗莎.jpg
> 像這類文件名我們需要截取一下字符串,只獲取名字這部分,而不需要獲取路徑部分的信息。
通過如下代碼對文件名進行截取字符串的操作:
if(name.contains("\\")){ //如果包含則截取字符串 name = name.substring(name.lastIndexOf("\\")+1); }
第二個問題
> 上傳的文件有可能出現重名,後上傳的文件會將先上傳的文件覆蓋。
> 解決:給文件名加一個唯一的前綴。
唯一標識_fennu.jpg
UUID_fennu.jpg
第三個問題
> 有些情況需要限制上傳文件的大小。
- 設置單個文件大小為50KB:
fileUpload.setFileSizeMax(1024*50);
- 設置完單個文件大小限制以後,一旦上傳的文件超過限制,則會拋出如下異常:
FileSizeLimitExceededException
所有可以對該異常進行捕獲,當出現該異常時則設置一個錯誤消息。
- 設置多個文件的總大小為150KB
fileUpload.setSizeMax(1024*150);
- 當多個文件的大小超出範圍時,會拋出如下異常
SizeLimitExceededException
第四個問題
> 當用戶上傳一個空的文件,依然會將文件保存到硬盤上。
> 在保存文件應該先對文件的大小進行判斷,如果size為0,則不處理。
2.文件的下載
[1] 簡介
> 將服務器中的文件下載到本地。
> 一般情況下資源所在的鏈接發送給瀏覽器,瀏覽器就會自動下載。
但是當瀏覽器支持當前文件的格式,瀏覽器會自動打開文件,而不會彈出下載窗口。
> 直接將資源放在項目的目錄下,瀏覽器可以直接訪問到資源。
所以一般我們下載的資源不能讓瀏覽器直接訪問到。
[2] 下載所需要的內容
1.獲取到文件的流
2.設置兩個響應頭
[3] 下載相關的兩個響應頭
1) 文件類型 Content-Type --> 文件的MIME類型
Content-Type:告訴瀏覽器文件的類型,需要設置一個MIME值
response.setContent-Type("MIME值")
通過servletContext.getMimeType(path)方法可以直接獲取文件的MIME類型
2) 下載文件的信息 Content-Disposition --> attachment; filename=文件名
Content-Disposition告訴瀏覽器如何處理文件,
attachment 告訴瀏覽器這個文件是一個附件的形式發給你的,需要你做下載的操作
filename 告訴瀏覽器下載文件的名字
3) 亂碼的問題,當將文件的名字設置為中文,瀏覽器正常顯示文件的名字。
因為從服務器向瀏覽器發送中文時,需要對內容進行URL編碼。
> 大部分瀏覽器使用如下方式即可解決亂碼問題:URLEncoder.encode(fileName, "utf-8");
> 但是火狐默認以Base64來解碼的,所以要為火狐單獨處理。
> 可以使用如下代碼來判斷瀏覽器的類型,然後進行不同的編碼處理
1 //判斷當前瀏覽器是否為火狐 2 3 if(ua.contains("Firefox")){ 4 5 //是火狐瀏覽器,使用BASE64編碼 6 7 fileName = "=?utf-8?b?"+new BASE64Encoder().encode(fileName.getBytes("utf-8"))+"?="; 8 9 }else{ 10 11 //給文件名進行URL編碼 12 13 //URLEncoder.encode()需要兩個參數,第一個參數時要編碼的字符串,第二個是編碼所采用的字符集 14 15 fileName = URLEncoder.encode(fileName, "utf-8"); 16 17 }
> 還有一種不太講理的方式,誰問跟誰急。反正好使
- 向將字符串用gbk進行解碼,然後在使用iso8859-1進行編碼
fileName = new String(fileName.getBytes("gbk"),"iso8859-1");
3.案例
[1]文件上傳:
(1)原生方法實現文件上傳
1 public void testUpload(CommonsMultipartFile file,HttpServletRequest request){ 2 //聲明輸入流和輸出流 3 InputStream in=null;OutputStream out=null; 4 //獲取ServletContext 5 ServletContext servletContext=request.getServletContext(); 6 //獲取文件目錄全路徑 7 String realPath = servletContext.getRealPath("/WEB-INF/upload"); 8 //獲取文件名 9 String fileName=file.getOriginalFilename(); 10 //聲明唯一標識,保證每次上傳保存的文件名不同,從而使每次上傳都可以成功,即使是相同的文件或資源 11 UUID id=UUID.randomUUID(); 12 //將唯一標識與原文件名相加生成新資源文件名 13 fileName=id+"_"+fileName; 14 //通過文件目錄全路徑新建一個文件對象 15 File file1=new File(realPath); 16 //判斷文件目錄是否存在 17 if(!file1.exists()){ 18 //如果不存在,則創建目錄文件夾 19 file1.mkdirs(); 20 21 } 22 23 try { 24 //設置輸入流與輸出流 25 in=file.getInputStream(); 27 out=new FileOutputStream(new File(realPath+"\\"+fileName)); 28 //設置緩沖區 29 byte[] buffer=new byte[1024]; 30 31 int len=0; 32 //判斷文件資源是否輸入完,如果輸入晚,則len==-1 33 while((len=in.read(buffer))!=-1){ 34 //通過輸出流寫入文件 35 out.write(buffer, 0, len); 36 37 } 38 //這是簡便方法直接可以將輸入流寫入輸出流 39 /*IOUtils.copy(in, out);*/ 40 41 } catch (IOException e) { 42 43 // TODO Auto-generated catch block 44 45 e.printStackTrace(); 46 47 }finally{ 48 //關閉輸入流 49 if(in!=null){ 50 51 try { 52 53 in.close(); 54 55 } catch (IOException e) { 56 57 // TODO Auto-generated catch block 58 59 e.printStackTrace(); 60 61 } 62 63 } 64 //關閉輸出流 65 if(out!=null){ 66 67 try { 68 69 out.close(); 70 71 } catch (IOException e) { 72 73 // TODO Auto-generated catch block 74 75 e.printStackTrace(); 76 77 } 78 79 } 80 81 } 82 83 }
(2)通過fileItem實現文件上傳
1 protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { 2 3 DiskFileItemFactory dFactory=new DiskFileItemFactory(); 4 5 ServletFileUpload sUpload=new ServletFileUpload(dFactory); 6 7 sUpload.setHeaderEncoding("UTF-8"); 8 9 try { 10 11 sUpload.setFileSizeMax(1024*50);//設置單個上傳文件的大小 12 13 sUpload.setSizeMax(1024*1024);//設置總上傳文件的大小 14 15 List<FileItem> list = sUpload.parseRequest(request); 16 17 for(FileItem f:list){ 18 19 if(f.isFormField()){ 20 21 String name = f.getFieldName(); 22 23 String string = f.getString(); 24 25 System.out.println(name+":"+string); 26 27 } 28 29 else { 30 31 //獲取文件名 32 33 String fileName = f.getName(); 34 35 if(fileName.contains("\\")){ 36 37 fileName = fileName.substring(fileName.lastIndexOf("\\")+1); 38 39 } 40 41 //獲取上傳路徑 42 43 String realPath = getServletContext().getRealPath("/WEB-INF/upload"); 44 45 //檢查upload文件夾是否存在,如果不存在則創建 46 47 System.out.println(realPath); 48 49 File file = new File(realPath); 50 51 if(!file.exists()){ 52 53 file.mkdirs(); 54 55 }; 56 57 //為避免重名生成一個uuid作為文件名的前綴 58 59 String prefix = UUID.randomUUID().toString().replace("-", ""); 60 61 //將文件寫入到服務器中 62 63 f.write(new File(realPath+"/"+prefix+"_"+fileName)); 64 65 //清楚文件緩存 66 67 f.delete(); 68 69 } 70 71 } 72 73 } catch(FileSizeLimitExceededException e){ 74 75 System.out.println("文件內容太大,無法上傳"); 76 77 }catch ( Exception e) { 78 79 // TODO Auto-generated catch block 80 81 e.printStackTrace(); 82 } 83 }
[2]文件下載:
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // TODO Auto-generated method stub ServletContext servletContext=request.getServletContext(); String fileName="風吹麥浪.mp3"; String path = servletContext.getRealPath("WEB-INF/"+fileName); File file=new File(path); String type = servletContext.getMimeType(path); InputStream inputStream=new FileInputStream(file); response.setContentType(type); fileName=new String(fileName.getBytes("gbk"),"iso8859-1"); response.setHeader("Content-Disposition","attachment;filename="+fileName); ServletOutputStream outputStream=response.getOutputStream(); IOUtils.copy(inputStream, outputStream); }
文件或資源的上傳和下載