知識
不管是網(wǎng)站,軟件還是小程序,都要直接或間接能為您產(chǎn)生價值,我們在追求其視覺表現(xiàn)的同時,更側(cè)重于功能的便捷,營銷的便利,運營的高效,讓網(wǎng)站成為營銷工具,讓軟件能切實提升企業(yè)內(nèi)部管理水平和效率。優(yōu)秀的程序為后期升級提供便捷的支持!
您當(dāng)前位置>首頁 » 新聞資訊 » 網(wǎng)站建設(shè) >
JavaWeb中(文件上傳和下載)
發(fā)表時間:2020-8-19
發(fā)布人:葵宇科技
瀏覽次數(shù):80
引言:
-
文件的上傳和下載在web應(yīng)用中是非常常用,也是非常有用的功能。
- 例如:發(fā)送電子郵件時可以同過上傳附件發(fā)送文件,OA系統(tǒng)中可以通過上傳文件來提交公文,社交網(wǎng)站通過上傳圖片來自定義頭像等等。
- 例如:下載實際上只要資源放在用戶可訪問的目錄中用戶就可以直接通過地址下載,但是一些資源是存放到數(shù)據(jù)庫中的,還有一些資源需要一定權(quán)限才能下載,這里就需要我們通過Servlet來完成下載的功能。
-
可以說上傳和下載是每一個web應(yīng)用都需要具有的一個功能,所以需要我們掌握。
-
第1章 文件的上傳
1.1 文件上傳的步驟
文件的上傳主要分成兩個步驟:
-
用戶在頁面中選擇要上傳的文件,然后將請求提交到Servlet
-
Servlet收到請求,解析用戶上傳的文件,然后將文件存儲到服務(wù)器
1.2 創(chuàng)建上傳文件的表單
- 創(chuàng)建一個form表單
<form action="" method="post" enctype="multipart/form-data">
<input type="file" name="file" /><br /><br />
<input type="submit" value="上傳" />
</form>
-
文件上傳的表單和之前的表單類似,但有以下內(nèi)容需要注意:
- 表單的method屬性必須為post
- 表單的enctype屬性必須為multipart/form-data
- 上傳文件的控件是input,type屬性為file
-
該表單打開后是如下效果:
-
IE
[外鏈圖片轉(zhuǎn)存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-0ohOyE7Q-1597829644379)(尚硅谷_張春勝_文件的上傳和下載.assets/1558975331009.png)]
-
Chrome
[外鏈圖片轉(zhuǎn)存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-bQ5iOiZO-1597829644383)(尚硅谷_張春勝_文件的上傳和下載.assets/1558975309963.png)]
-
火狐
[外鏈圖片轉(zhuǎn)存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-VYT1AJDN-1597829644385)(尚硅谷_張春勝_文件的上傳和下載.assets/1558975370024.png)]
-
-
編寫Servelet
-
頁面的表單控件創(chuàng)建好以后,選中文件點擊上傳按鈕請求將會提交到指定的Servlet來處理。
-
注意:這里不能再像以前的Servlet中那樣,通過request.getParamter()來獲取請求參數(shù)了,當(dāng)enctype=“multipart/form-data” 時,再使用getParamter()獲取到內(nèi)容永遠(yuǎn)為空。因為瀏覽器發(fā)送請求的方式已經(jīng)改變。
-
既然以前的方法不能使用了,這里我們必須要引入一個新的工具來解析請求中的參數(shù)和文件,這個工具就是commons-fileupload。
-
1.3 commons-fileupload
-
commons-fileupload是Apache開發(fā)的一款專門用來處理上傳的工具,它的作用就是可以從request對象中解析出,用戶發(fā)送的請求參數(shù)和上傳文件的流。
-
commons-fileupload包依賴commons-io,兩個包需要同時導(dǎo)入。
-
核心類:
-
DiskFileItemFactory
-
工廠類,用于創(chuàng)建ServletFileUpload,設(shè)置緩存等
-
該類一般直接使用構(gòu)造器直接創(chuàng)建實例
-
方法:
-
public void setSizeThreshold(int sizeThreshold):用于設(shè)置緩存文件的大小(默認(rèn)值10kb)
-
public void setRepository(File repository):用于設(shè)置緩存文件位置(默認(rèn)系統(tǒng)緩存目錄)
-
-
-
ServletFileUpload
-
該類用于解析request對象從而獲取用戶發(fā)送的請求參數(shù)(包括普通參數(shù)和文件參數(shù))
-
該類需要調(diào)用有參構(gòu)造器創(chuàng)建實例,構(gòu)造器中需要一個DiskFileItemFactory作為參數(shù)
-
方法:
-
public List parseRequest(HttpServletRequest request):解析request對象,獲取請求參數(shù),返回的是一個List,List中保存的是一個FileItem對象,一個對象代表一個請求參數(shù)。
-
public void setFileSizeMax(long fileSizeMax):設(shè)置單個文件的大小限制,單位為B。如果上傳文件超出限制,會在parseRequest()拋出異常FileSizeLimitExceededException。
-
public void setSizeMax(long sizeMax):限制請求內(nèi)容的總大小,單位為B。如果上傳文件超出限制,會在parseRequest()拋出異常SizeLimitExceededException。
-
-
-
FileItem
-
該類用于封裝用戶發(fā)送的參數(shù)和文件,也就是用戶發(fā)送來的信息將會被封裝成一個FileItem對象,我們通過該對象獲取請求參數(shù)或上傳文件的信息。
-
該類不用我們手動創(chuàng)建,由ServletFileItem解析request后返回。
-
方法:
-
String getFieldName():獲取表單項的名字,也就是input當(dāng)中的name屬性的值。
-
String getName():獲取上傳的文件名,普通的請求參數(shù)為null。
-
String getString(String encoding):獲取內(nèi)容,encoding參數(shù)需要指定一個字符集。
? ① 若為文件,將文件的流轉(zhuǎn)換為字符串。
? ② 若為請求參數(shù),則獲取請求參數(shù)的value。
-
boolean isFormField():判斷當(dāng)前的FileItem封裝的是普通請求參數(shù),還是一個文件。
? ① 如果為普通參數(shù)返回:true
? ② 如果為文件參數(shù)返回:false
-
String getContentType():獲取上傳文件的MIME類型
-
long getSize():獲取內(nèi)容的大小
-
write():將文件上傳到服務(wù)器
-
-
-
-
示例代碼:創(chuàng)建一個Servlet并在doPost()方法中編寫如下代碼
//創(chuàng)建工廠類 DiskFileItemFactory factory = new DiskFileItemFactory(); //創(chuàng)建請求解析器 ServletFileUpload fileUpload = new ServletFileUpload(factory); //設(shè)置上傳單個文件的的大小 fileUpload.setFileSizeMax(1024*1024*3); //設(shè)置上傳總文件的大小 fileUpload.setSizeMax(1024*1024*3*10); //設(shè)置響應(yīng)內(nèi)容的編碼 response.setContentType("text/html;charset=utf-8"); try { //解析請求信息,獲取FileItem的集合 List<FileItem> items = fileUpload.parseRequest(request); //遍歷集合 for (FileItem fileItem : items) { //如果是普通的表單項 if(fileItem.isFormField()){ //獲取參數(shù)名 String fieldName = fileItem.getFieldName(); //獲取參數(shù)值 String value = fileItem.getString("utf-8"); System.out.println(fieldName+" = "+value); //如果是文件表單項 }else{ //獲取文件名 String fileName = fileItem.getName(); //獲取上傳路徑 String realPath = getServletContext().getRealPath("/WEB-INF/upload"); //檢查upload文件夾是否存在,如果不存在則創(chuàng)建 File f = new File(realPath); if(!f.exists()){ f.mkdir(); }; //為避免重名生成一個uuid作為文件名的前綴 String prefix = UUID.randomUUID().toString().replace("-", ""); //將文件寫入到服務(wù)器中 fileItem.write(new File(realPath+"/"+prefix+"_"+fileName)); //清楚文件緩存 fileItem.delete(); } } } catch (Exception e) { if(e instanceof SizeLimitExceededException){ //文件總大小超出限制 response.getWriter().print("上傳文件的總大小不能超過30M"); }else if(e instanceof FileSizeLimitExceededException){ //單個文件大小超出限制 response.getWriter().print("上傳單個文件的大小不能超過3M"); } } response.getWriter().print("上傳成功");
第2章 文件的下載
2.1 使用說明
-
文件下載最直接的方法就是把文件直接放到服務(wù)器的目錄中,用戶直接訪問該文件就可以直接下載。
-
但是實際上這種方式并不一定好用,比如我們在服務(wù)器上直接放置一個MP3文件,然后通過瀏覽器訪問該文件的地址,如果是IE瀏覽器可能就會彈出下載窗口,而如果是FireFox和Chrome則有可能直接播放。再有就是有一些文件我們是不希望用戶可以直接訪問到的,這是我們就要通過Servlet來完成下載功能。
-
下載文件的關(guān)鍵是幾點:
-
服務(wù)器以一個流的形式將文件發(fā)送給瀏覽器。
-
發(fā)送流的同時還需要設(shè)置幾個響應(yīng)頭,來告訴瀏覽器下載的信息。
- 具體響應(yīng)頭如下:
- Content-Type
- 下載文件的MIME類型
- 可以通過servletContext. getMimeType(String file)獲取
- 也可以直接手動指定
- 使用response.setContentType(String type);
- 響應(yīng)頭樣式:Content-Type: audio/mpeg
- Content-Disposition
- 下載文件的名字,主要作用是提供一個默認(rèn)的用戶名
- 通過response.setHeader(“Content-Disposition”, disposition)設(shè)置
- 響應(yīng)頭樣式:Content-Disposition: attachment; filename=xxx.mp3
- Content-Length
- 下載文件的長度,用于設(shè)置文件的長處(不必須)
- 通過response. setContentLength(int len)設(shè)置。
- 設(shè)置后樣式:Content-Length: 3140995
- Content-Type
- 具體響應(yīng)頭如下:
-
接下來需要以輸入流的形式讀入硬盤上的文件
- FileInputStream is = new FileInputStream(file);
- 這個流就是我們一會要發(fā)送給瀏覽器的內(nèi)容
-
通過response獲取一個輸出流,并將文件(輸入流)通過該流發(fā)送給瀏覽器
-
獲取輸出流:ServletOutputStream out = response.getOutputStream();
-
通過輸出流向瀏覽器發(fā)送文件(不要忘了關(guān)閉輸入流)
byte[] b = new byte[1024]; int len = 0; while((len=is.read(b))> 0){ out.write(b, 0, len); } is.close();
-
-
2.2 步驟演示
-
一下步驟都是在同一個Servlet的doGet()方法中編寫的
-
我所下載的文件是放在WEB-INF下mp3文件夾中的文件
-
具體步驟
-
獲取文件的流:
String realPath = getServletContext().getRealPath("/WEB-INF/mp3/中國話.mp3"); //獲取文件的File對象 File file = new File(realPath); //獲取文件的輸入流 FileInputStream is = new FileInputStream(file);
-
獲取頭信息:
//獲取文件的MIME信息 String contentType = getServletContext().getMimeType(realPath); //設(shè)置下載文件的名字 String filename = "zhongguohua.mp3"; //創(chuàng)建Content-Disposition信息 String disposition = "attachment; filename="+ filename ; //獲取文件長度 long size = file.length();
-
設(shè)置頭信息
//設(shè)置Content-Type response.setContentType(contentType); //設(shè)置Content-Disposition response.setHeader("Content-Disposition", disposition); //設(shè)置文件長度 response.setContentLength((int)size);
-
發(fā)送文件
//通過response獲取輸出流,用于向瀏覽器輸出內(nèi)容 ServletOutputStream out = response.getOutputStream(); //將文件輸入流通過輸出流輸出 byte[] b = new byte[1024]; int len = 0; while((len=is.read(b))> 0){ out.write(b, 0, len); } //最后不要忘記關(guān)閉輸入流,輸出流由Tomcat自己處理,我們不用手動關(guān)閉 is.close();
-
2.3 亂碼
-
至此實際上文件下載的主要功能都已經(jīng)完成。但是還有一個問題我們這里沒有體現(xiàn)出來,因為目前我們的文件名使用的是純英文的,沒有亂碼問題。這里如果我們要使用中文文件名的話,毫無疑問會出現(xiàn)亂碼問題。
-
解決此問題的方法很簡單,在獲取文件名之后為文件名進行編碼:
filename = java.net.URLEncoder.encode(filename,"utf-8");
-
但是注意這里火狐瀏覽器比較特殊,因為他默認(rèn)是以BASE64解碼的,所以這塊如果需要考慮火狐的問題的話還需要特殊處理一下。
- 先要獲取客戶端信息(通過獲取請求頭中的User-Agent信息)
//獲取客戶端信息 String ua = request.getHeader("User-Agent");
- 然后判斷瀏覽器版本,做不同的處理(通過判斷頭信息中是否包含F(xiàn)irefox字符串來判斷瀏覽器版本)
//判斷客戶端是否為火狐 if(ua.contains("Firefox")){ //若為火狐使用BASE64編碼 filename = "=?utf-8?B?"+new BASE64Encoder() .encode(filename.getBytes("utf-8"))+"?="; }else{ //否則使用UTF-8 filename = URLEncoder.encode(filename,"utf-8"); }
完整代碼:
//設(shè)置響應(yīng)字符集
response.setContentType("text/html; charset=UTF-8");
DiskFileItemFactory diskFileItemFactory=new DiskFileItemFactory();
//創(chuàng)建解析器
ServletFileUpload servletFileUpload=new ServletFileUpload(diskFileItemFactory);
//設(shè)置上傳文件的大小
servletFileUpload.setFileSizeMax(1024*500);
//獲取上傳的真實路徑
String realPath = getServletContext().getRealPath("/upload");
//優(yōu)化1:如果文件路勁不存在,則重新創(chuàng)建一個
File refile=new File(realPath);
if(refile.exists()==false){
refile.mkdirs();
}
//優(yōu)化2:若同名文件 則添加不上,因此解決同名文件,讓其添加上。
//優(yōu)化3:設(shè)置上傳文件大小
//在前臺傳來的文件雖然為同一個文件,但是在后臺我們可以將文件名進行擴展。
// 通過ServletFileUpload中的List<FileItem> parseRequest(request),將request解析為List<FileItem>
try {
List<FileItem> fileItems = servletFileUpload.parseRequest(request);
//遍歷集合
for (FileItem fileItem : fileItems) {
if(fileItem.isFormField()==false){
//獲取文件名
String fname = fileItem.getName();
//處理文件名
String replacename = UUID.randomUUID().toString().replace("-", "");
//將文件上傳到服務(wù)器的指定位置
File file=new File(realPath+File.separator+replacename+fname);
fileItem.write(file);
PrintWriter writer = response.getWriter();
writer.write("upload success");
}
}
}
catch (FileUploadBase.FileSizeLimitExceededException e){
response.getWriter().write("上傳文件大小超過50KB");
}
catch (Exception e) {
e.printStackTrace();
}
jsp代碼:
<%--
Created by IntelliJ IDEA.
User: admin
Date: 2020/8/19
Time: 16:29
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>下載demo</title>
</head>
<body>
<a href="DownServlet?fname=gson.jar">jar包下載</a>
<a href="DownServlet?fname=a698e5967c4d4c23a895ce42e3289d260.jpg">千庫網(wǎng)_粉色背景里的玫瑰花束_攝影圖編號73147.jpg</a>
<a href="">xxx.pptx</a>
</body>
</html>
相關(guān)案例查看更多
相關(guān)閱讀
- 公眾號模板消息
- 云南網(wǎng)站建設(shè)公司地址
- 網(wǎng)站建設(shè)制作
- 楚雄網(wǎng)站建設(shè)公司
- 云南網(wǎng)站建設(shè)列表網(wǎng)
- 百度快速排名
- 大理網(wǎng)站建設(shè)公司
- php網(wǎng)站
- 二叉樹
- 云南網(wǎng)站制作哪家好
- 模版消息
- 紅河小程序開發(fā)
- 小程序表單
- 保險網(wǎng)站建設(shè)公司
- 網(wǎng)站建設(shè)首選
- 小程序
- 網(wǎng)站建設(shè)需要多少錢
- 網(wǎng)站開發(fā)公司哪家好
- 河南小程序制作
- 云南省建設(shè)廳網(wǎng)站官網(wǎng)
- 云南網(wǎng)站建設(shè)費用
- 大理小程序開發(fā)
- 云南做百度小程序的公司
- painter
- 云南網(wǎng)站優(yōu)化公司
- 迪慶小程序開發(fā)
- 汽車報廢回收管理系統(tǒng)
- 云南網(wǎng)站建設(shè)制作
- 云南小程序定制
- 網(wǎng)站排名