Web服務器收到客戶端的http請求,會針對每一次請求,分別創建一個用于代表請求的request對象、和代表響應的response對象。request和response對象即然代表請求和響應,那我們要獲取客戶機提交過來的數據,只需要找request對象就行了。要向客戶機輸出數據,只需要找response對象就行了。
一、HttpServletResponse對象介紹
HttpServletResponse對象代表服務器的響應。這個對象中封裝了向客戶端發送數據、發送響應頭,發送響應狀態碼的方法。查看HttpServletResponse的API,可以看到這些相關的方法。
1.1、負責向客戶端(瀏覽器)發送數據的相關方法
1.2、負責向客戶端(瀏覽器)發送響應頭的相關方法
1.3、負責向客戶端(瀏覽器)發送響應狀態碼的相關方法
1.4、響應狀態碼的常量
HttpServletResponse定義了很多狀態碼的常量(具體可以查看Servlet的API),當需要向客戶端發送響應狀態碼時,可以使用這些常量,避免了直接寫數字,常見的狀態碼對應的常量:
-
狀態碼404對應的常量
-
狀態碼200對應的常量
-
狀態碼500對應的常量
二、HttpServletResponse對象常見應用
2.1、使用OutputStream流向客戶端瀏覽器輸出中文數據
使用OutputStream流輸出中文注意問題:
<font color="red">在服務器端,數據是以哪個碼表輸出的,那么就要控制客戶端瀏覽器以相應的碼表打開,比如:outputStream.write("中國".getBytes("UTF-8"))
;使用OutputStream流向客戶端瀏覽器輸出中文,以UTF-8
的編碼進行輸出,此時就要控制客戶端瀏覽器以UTF-8
的編碼打開,否則顯示的時候就會出現中文亂碼,那么在服務器端如何控制客戶端瀏覽器以以UTF-8的編碼顯示數據呢?可以通過設置響應頭控制瀏覽器的行為,例如:response.setHeader("content-type", "text/html;charset=UTF-8")
;通過設置響應頭控制瀏覽器以UTF-8
的編碼顯示數據。</font>
范例:使用OutputStream流向客戶端瀏覽器輸出"中國"這兩個漢字
package gacl.response.study;import java.io.IOException;
import java.io.OutputStream;
import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class ResponseDemo01 extends HttpServlet { private static final long serialVersionUID = 4312868947607181532L; public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { outputChineseByOutputStream(response);//使用OutputStream流輸出中文 } /** * 使用OutputStream流輸出中文 * @param request * @param response * @throws IOException */ public void outputChineseByOutputStream(HttpServletResponse response) throws IOException{ /**使用OutputStream輸出中文注意問題: * 在服務器端,數據是以哪個碼表輸出的,那么就要控制客戶端瀏覽器以相應的碼表打開, * 比如:outputStream.write("中國".getBytes("UTF-8"));//使用OutputStream流向客戶端瀏覽器輸出中文,以UTF-8的編碼進行輸出 * 此時就要控制客戶端瀏覽器以UTF-8的編碼打開,否則顯示的時候就會出現中文亂碼,那么在服務器端如何控制客戶端瀏覽器以以UTF-8的編碼顯示數據呢? * 可以通過設置響應頭控制瀏覽器的行為,例如: * response.setHeader("content-type", "text/html;charset=UTF-8");//通過設置響應頭控制瀏覽器以UTF-8的編碼顯示數據 */ String data = "中國"; OutputStream outputStream = response.getOutputStream();//獲取OutputStream輸出流 response.setHeader("content-type", "text/html;charset=UTF-8");//通過設置響應頭控制瀏覽器以UTF-8的編碼顯示數據,如果不加這句話,那么瀏覽器顯示的將是亂碼 /** * data.getBytes()是一個將字符轉換成字節數組的過程,這個過程中一定會去查碼表, * 如果是中文的操作系統環境,默認就是查找查GB2312的碼表, * 將字符轉換成字節數組的過程就是將中文字符轉換成GB2312的碼表上對應的數字 * 比如: "中"在GB2312的碼表上對應的數字是98 * "國"在GB2312的碼表上對應的數字是99 */ /** * getBytes()方法如果不帶參數,那么就會根據操作系統的語言環境來選擇轉換碼表,如果是中文操作系統,那么就使用GB2312的碼表 */ byte[] dataByteArr = data.getBytes("UTF-8");//將字符轉換成字節數組,指定以UTF-8編碼進行轉換 outputStream.write(dataByteArr);//使用OutputStream流向客戶端輸出字節數組 } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
運行結果如下:
客戶端瀏覽器接收到數據后,就按照響應頭上設置的字符編碼來解析數據,如下所示:
2.2、使用PrintWriter流向客戶端瀏覽器輸出中文數據
使用PrintWriter流輸出中文注意問題:
<font color="red"> 在獲取PrintWriter輸出流之前首先使用"response.setCharacterEncoding(charset)
"設置字符以什么樣的編碼輸出到瀏覽器,如:response.setCharacterEncoding("UTF-8")
;設置將字符以"UTF-8"編碼輸出到客戶端瀏覽器,然后再使用response.getWriter()
;獲取PrintWriter輸出流</font>. 這兩個步驟不能顛倒,如下:
response.setCharacterEncoding("UTF-8");//設置將字符以"UTF-8"編碼輸出到客戶端瀏覽器
/*** PrintWriter out = response.getWriter();這句代碼必須放在response.setCharacterEncoding("UTF-8");之后 * 否則response.setCharacterEncoding("UTF-8")這行代碼的設置將無效,瀏覽器顯示的時候還是亂碼 */ PrintWriter out = response.getWriter();//獲取PrintWriter輸出流
然后再使用response.setHeader("content-type", "text/html;charset=字符編碼")
;設置響應頭,控制瀏覽器以指定的字符編碼編碼進行顯示,例如:
//通過設置響應頭控制瀏覽器以UTF-8的編碼顯示數據,如果不加這句話,那么瀏覽器顯示的將是亂碼response.setHeader("content-type", "text/html;charset=UTF-8");
上述兩步可以合成一步完成:
response.setContentType("text/html; charset=UTF-8");
除了可以使用response.setHeader("content-type", "text/html;charset=字符編碼")
;設置響應頭來控制瀏覽器以指定的字符編碼編碼進行顯示這種方式之外,還可以用如下的方式來模擬響應頭的作用.
/*** 多學一招:使用HTML語言里面的<meta>標簽來控制瀏覽器行為,模擬通過設置響應頭控制瀏覽器行為*response.getWriter().write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'/>"); * 等同于response.setHeader("content-type", "text/html;charset=UTF-8"); */ response.getWriter().write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'/>");
范例:使用PrintWriter流向客戶端瀏覽器輸出"中國"這兩個漢字
package gacl.response.study;import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class ResponseDemo01 extends HttpServlet { private static final long serialVersionUID = 4312868947607181532L; public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { outputChineseByPrintWriter(response);//使用PrintWriter流輸出中文 } /** * 使用PrintWriter流輸出中文 * @param request * @param response * @throws IOException */ public void outputChineseByPrintWriter(HttpServletResponse response) throws IOException{ String data = "中國"; //通過設置響應頭控制瀏覽器以UTF-8的編碼顯示數據,如果不加這句話,那么瀏覽器顯示的將是亂碼 //response.setHeader("content-type", "text/html;charset=UTF-8"); response.setCharacterEncoding("UTF-8");//設置將字符以"UTF-8"編碼輸出到客戶端瀏覽器 /** * PrintWriter out = response.getWriter();這句代碼必須放在response.setCharacterEncoding("UTF-8");之后 * 否則response.setCharacterEncoding("UTF-8")這行代碼的設置將無效,瀏覽器顯示的時候還是亂碼 */ PrintWriter out = response.getWriter();//獲取PrintWriter輸出流 /** * 多學一招:使用HTML語言里面的<meta>標簽來控制瀏覽器行為,模擬通過設置響應頭控制瀏覽器行為 * out.write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'/>"); * 等同于response.setHeader("content-type", "text/html;charset=UTF-8"); */ out.write("<meta http-equiv='content-type' content='text/html;charset=UTF-8'/>"); out.write(data);//使用PrintWriter流向客戶端輸出字符 } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
當需要向瀏覽器輸出字符數據時,使用PrintWriter比較方便,省去了將字符轉換成字節數組那一步。
2.3、使用OutputStream或者PrintWriter向客戶端瀏覽器輸出數字
比如有如下的代碼:
package gacl.response.study;import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintWriter; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class ResponseDemo01 extends HttpServlet { private static final long serialVersionUID = 4312868947607181532L; public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { outputOneByOutputStream(response);//使用OutputStream輸出1到客戶端瀏覽器 } /** * 使用OutputStream流輸出數字1 * @param request * @param response * @throws IOException */ public void outputOneByOutputStream(HttpServletResponse response) throws IOException{ response.setHeader("content-type", "text/html;charset=UTF-8"); OutputStream outputStream = response.getOutputStream(); outputStream.write("使用OutputStream流輸出數字1:".getBytes("UTF-8")); outputStream.write(1); } }
運行上面代碼顯示的結果如下:
運行的結果和我們想象中的不一樣,數字1沒有輸出來,下面我們修改一下上面的outputOneByOutputStream
方法的代碼,修改后的代碼如下:
/*** 使用OutputStream流輸出數字1* @param request* @param response* @throws IOException */public void outputOneByOutputStream(HttpServletResponse response) throws IOException{ response.setHeader("content-type", "text/html;charset=UTF-8"); OutputStream outputStream = response.getOutputStream(); outputStream.write("使用OutputStream流輸出數字1:".getBytes("UTF-8")); //outputStream.write(1); outputStream.write((1+"").getBytes()); }
1+""
這一步是將數字1和一個空字符串相加,這樣處理之后,數字1就變成了字符串1了,然后再將字符串1轉換成字節數組使用OutputStream
進行輸出,此時看到的結果如下:
這次可以看到輸出來的1了,這說明了一個問題:在開發過程中,如果希望服務器輸出什么瀏覽器就能看到什么,那么在服務器端都要以字符串的形式進行輸出。
如果使用PrintWriter流輸出數字,那么也要先將數字轉換成字符串后再輸出,如下:
/*** 使用PrintWriter流輸出數字1* @param request* @param response* @throws IOException */public void outputOneByPrintWriter(HttpServletResponse response) throws IOException{ response.setHeader("content-type", "text/html;charset=UTF-8"); response.setCharacterEncoding("UTF-8"); PrintWriter out = response.getWriter();//獲取PrintWriter輸出流 out.write("使用PrintWriter流輸出數字1:"); out.write(1+""); }
2.4、文件下載
文件下載功能是web開發中經常使用到的功能,使用HttpServletResponse
對象就可以實現文件的下載
文件下載功能的實現思路:
1.獲取要下載的文件的絕對路徑
2.獲取要下載的文件名
3.設置content-disposition響應頭控制瀏覽器以下載的形式打開文件
4.獲取要下載的文件輸入流
5.創建數據緩沖區
6.通過response對象獲取OutputStream流
7.將FileInputStream流寫入到buffer緩沖區
8.使用OutputStream將緩沖區的數據輸出到客戶端瀏覽器
范例:使用Response實現文件下載
package gacl.response.study;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.PrintWriter; import java.net.URLEncoder; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * @author gacl * 文件下載 */ public class ResponseDemo02 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { downloadFileByOutputStream(response);//下載文件,通過OutputStream流 } /** * 下載文件,通過OutputStream流 * @param response * @throws FileNotFoundException * @throws IOException */ private void downloadFileByOutputStream(HttpServletResponse response) throws FileNotFoundException, IOException { //1.獲取要下載的文件的絕對路徑 String realPath = this.getServletContext().getRealPath("/download/1.JPG"); //2.獲取要下載的文件名 String fileName = realPath.substring(realPath.lastIndexOf("\\")+1); //3.設置content-disposition響應頭控制瀏覽器以下載的形式打開文件 response.setHeader("content-disposition", "attachment;filename="+fileName); //4.獲取要下載的文件輸入流 InputStream in = new FileInputStream(realPath); int len = 0; //5.創建數據緩沖區 byte[] buffer = new byte[1024]; //6.通過response對象獲取OutputStream流 OutputStream out = response.getOutputStream(); //7.將FileInputStream流寫入到buffer緩沖區 while ((len = in.read(buffer)) > 0) { //8.使用OutputStream將緩沖區的數據輸出到客戶端瀏覽器 out.write(buffer,0,len); } in.close(); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
運行結果如下所示:
范例:使用Response實現中文文件下載
下載中文文件時,需要注意的地方就是中文文件名要使用URLEncoder.encode
方法進行編碼(URLEncoder.encode(fileName, "字符編碼"))
,否則會出現文件名亂碼。
package gacl.response.study;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.PrintWriter; import java.net.URLEncoder; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * @author gacl * 文件下載 */ public class ResponseDemo02 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { downloadChineseFileByOutputStream(response);//下載中文文件 } /** * 下載中文文件,中文文件下載時,文件名要經過URL編碼,否則會出現文件名亂碼 * @param response * @throws FileNotFoundException * @throws IOException */ private void downloadChineseFileByOutputStream(HttpServletResponse response) throws FileNotFoundException, IOException { String realPath = this.getServletContext().getRealPath("/download/張家界國家森林公園.JPG");//獲取要下載的文件的絕對路徑 String fileName = realPath.substring(realPath.lastIndexOf("\\")+1);//獲取要下載的文件名 //設置content-disposition響應頭控制瀏覽器以下載的形式打開文件,中文文件名要使用URLEncoder.encode方法進行編碼,否則會出現文件名亂碼 response.setHeader("content-disposition", "attachment;filename="+URLEncoder.encode(fileName, "UTF-8")); InputStream in = new FileInputStream(realPath);//獲取文件輸入流 int len = 0; byte[] buffer = new byte[1024]; OutputStream out = response.getOutputStream(); while ((len = in.read(buffer)) > 0) { out.write(buffer,0,len);//將緩沖區的數據輸出到客戶端瀏覽器 } in.close(); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
運行結果如下所示:
文件下載注意事項:編寫文件下載功能時推薦使用OutputStream
流,避免使用PrintWriter
流,因為OutputStream
流是字節流,可以處理任意類型的數據,而PrintWriter
流是字符流,只能處理字符數據,如果用字符流處理字節數據,會導致數據丟失。
范例:使用PrintWriter流下載文件
package gacl.response.study;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.PrintWriter; import java.net.URLEncoder; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * @author gacl * 文件下載 */ public class ResponseDemo02 extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { downloadFileByPrintWriter(response);//下載文件,通過PrintWriter流 } /** * 下載文件,通過PrintWriter流,雖然也能夠實現下載,但是會導致數據丟失,因此不推薦使用PrintWriter流下載文件 * @param response * @throws FileNotFoundException * @throws IOException */ private void downloadFileByPrintWriter(HttpServletResponse response) throws FileNotFoundException, IOException { String realPath = this.getServletContext().getRealPath("/download/張家界國家森林公園.JPG");//獲取要下載的文件的絕對路徑 String fileName = realPath.substring(realPath.lastIndexOf("\\")+1);//獲取要下載的文件名 //設置content-disposition響應頭控制瀏覽器以下載的形式打開文件,中文文件名要使用URLEncoder.encode方法進行編碼 response.setHeader("content-disposition", "attachment;filename="+URLEncoder.encode(fileName, "UTF-8")); FileReader in = new FileReader(realPath); int len = 0; char[] buffer = new char[1024]; PrintWriter out = response.getWriter(); while ((len = in.read(buffer)) > 0) { out.write(buffer,0,len);//將緩沖區的數據輸出到客戶端瀏覽器 } in.close(); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } }
運行結果如下:
正常彈出下載框,此時我們點擊【保存】按鈕將文件下載下來,如下所示:
可以看到,只下載了5.25MB,而這張圖片的原始大小卻是
這說明在下載的時候數據丟失了,所以下載不完全,所以這張圖片雖然能夠正常下載下來,但是卻是無法打開的,因為丟失掉了部分數據,如下所示:
所以使用PrintWriter流處理字節數據,會導致數據丟失,這一點千萬要注意,因此在編寫下載文件功能時,要使用OutputStream流,避免使用PrintWriter流,因為OutputStream流是字節流,可以處理任意類型的數據,而PrintWriter流是字符流,只能處理字符數據,如果用字符流處理字節數據,會導致數據丟失。
?
from:?https://segmentfault.com/a/1190000004113284