1. 程式人生 > 實用技巧 >來吧,教你JavaWeb中如何實現檔案下載

來吧,教你JavaWeb中如何實現檔案下載

本文作者:樂位元組-坑王老薛

檔案下載的方式

  • 超連結下載
  • 後臺實現下載

案例實操

超連結下載

當我們在 HTML 或 JSP 頁面中使用標籤時,原意是希望能夠進行跳轉,但當超連結遇到瀏覽器不識別的動態網頁時則會自動下載。如果瀏覽器遇到能夠直接顯示的資源,瀏覽器就會預設顯示出來,比如 txt,png,jpg 等。當然我們也可以通過 download 屬性規定瀏覽器進行下載。但有些瀏覽器並不支援。

預設下載

<a href="upload/abc.zip">超連結下載</a>

指定 download 屬性下載

<a href="upload/abc.txt" download="abcdef.txt">超連結下載</a>

這裡,download 也可以不寫任何資訊,會自動使用預設檔名。這樣當用戶開啟瀏覽器點選連結的時候就會直接下載檔案。

後臺實現下載

Step1:需要通過 HttpServletResponse.setContentType 方法設定 Content-type 頭欄位的值,這樣瀏覽器才能夠使用某種方式或啟用某個程式來處理相應 MIME 型別的資料,例 如 ”application/octet-stream” 或 ”application/x-msdownload” 等

Step2:需要通過 HttpServletResponse.setHeader 方法設定 Content-Disposition 頭的值為”attachment;filename=檔名”,filename提供了檔案下載時的一個預設檔名

Step3:讀取下載檔案,呼叫 HttpServletResponse.getOutputStream 方法返回的OutputStream物件來向客戶端寫入附件內容。

public class DownLoadServlet extends HttpServlet {
	private static final long serialVersionUID = 1L;

	protected void service(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		
		// 設定請求編碼
		request.setCharacterEncoding("UTF-8");
		// 接受引數,得到需要被下載的檔案的名稱
		String fileName = request.getParameter("fileName");
		// 判斷名字名是否為空
		if(fileName == null || "".equals(fileName)){
			// 提示
			System.out.println("檔名不能為空");
			return;
		}
		// 獲取檔案存放的真實路徑
		String path = request.getServletContext().getRealPath("/" + fileName);
		// 通過檔案路徑和檔名得到file物件
		File file = new File(path);
		// 判斷是否存在,並且是一個標準檔案
		if (file.exists() && file.isFile()){
			// 設定相應型別 application/octet-stream
			response.setContentType("application/x-msdownload");
			// 設定頭資訊
			response.setHeader("Content-Disposition", "attachment;filename=" + fileName);
			// 通過file物件得到輸入流
			FileInputStream in = new FileInputStream(file);
			// 得到輸出流
			ServletOutputStream out = response.getOutputStream();
			byte[] car = new byte[1024];
			int len = 0;
			while((len = in.read(car)) != -1){
				out.write(car,0,len);
			}
			// 關閉流
			in.close();
			out.close();
		} else {
			System.out.println("檔案路徑不正確!");
		}
		
	}

}

擴充套件~HTML表單編碼

HTML表單編碼

enctype 屬性指定瀏覽器如何編碼資料並將其呈現給伺服器。

此屬性有三個允許值。

  • application/x-www-form-urlencoded
    預設編碼。
    此編碼無法用於將檔案上傳到伺服器。
  • multipart/form-data
    此編碼用於將檔案上傳到伺服器。
  • text/plain
    此編碼因瀏覽器而異。

要理解不同編碼的工作原理,我們建立了以下形式。

<!DOCTYPE HTML>
<html>
<body>
  <form method="post" action="http://example.com/form">
    <input name="fave" /> 
    <input name="name" />
    <button>Submit Vote</button>
  </form>
</body>
</html>

application/x-www-form-urlencoded

如果使用application / x-www-form-urlencoded編碼,每個資料項的名稱和值都使用用於編碼URL的相同方案進行編碼。這是編碼應用於示例形式的資料的方式:

fave=Apples&name=FiratName+LastName

特殊字元將替換為其HTML實體對應部分。資料項的名稱和值由等號(=)分隔,資料/值元組由&符號(&)分隔。

multipart/form-data

multipart / form-data 編碼往往僅用於上傳檔案。下面是示例表單中的資料如何編碼:

------WebKitFormBoundary2desQWER543CDFGF

Content-Disposition: form-data; name="fave" YourName
------WebKitFormBoundary2desQWER543CDFGF Content-Disposition: form-data; name="name" www.lezijie.cn
------WebKitFormBoundary2desQWER543CDFGF-- fave=Apple
name=www.lezijie.cn

multipart/plain

主流瀏覽器以不同的方式對該編碼進行編碼。

Google Chrome以與application / x-www-form-urlencoded方案相同的方式對資料進行編碼,而Firefox對資料進行編碼的方式如下:

fave=xml
name=www.lezijie.cn

n

主流瀏覽器以不同的方式對該編碼進行編碼。

Google Chrome以與application / x-www-form-urlencoded方案相同的方式對資料進行編碼,而Firefox對資料進行編碼的方式如下:

fave=xml
name=www.lezijie.cn

每個資料項都放在一行上,不會對特殊字元進行編碼。