1. 程式人生 > >javaWeb如何實現請求、響應?

javaWeb如何實現請求、響應?

先來看一個流程圖:

https://img-blog.csdn.net/20160815205011516?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center

伺服器處理請求的流程:

(1)伺服器每次收到請求時,都會為這個請求開闢一個新的執行緒。

(2)伺服器會把客戶端的請求資料封裝到request物件中,request就是請求資料的載體!

(3)伺服器還會建立response物件,這個物件與客戶端連線在一起,它可以用來向客戶端傳送響應。

由流程圖可以看出,在JavaWeb的請求與響應中,最重要的兩個引數為request以及response,這兩引數在Servlet的service( )方法中。

1、response概念:

response是Servlet.service方法的一個引數,型別為javax.servlet.http.HttpServletResponse。在客戶端發出每個請求時,伺服器都會建立一個response物件,並傳入給Servlet.service()方法。response物件是用來對客戶端進行響應的,這說明在service()方法中使用response物件可以完成對客戶端的響應工作。

response物件的功能分為以下四種:

(1)設定響應頭資訊

(2)傳送狀態碼

(3)設定響應正文

(4)重定向

2、response響應正文

response是響應物件,向客戶端輸出響應正文(響應體)可以使用response的響應流,repsonse一共提供了兩個響應流物件:

(1)PrintWriter out = response.getWriter():獲取字元流;

(2)ServletOutputStream out = response.getOutputStream():獲取位元組流;

當然,如果響應正文內容為字元,那麼使用response.getWriter(),如果響應內容是位元組,例如下載時,那麼可以使用response.getOutputStream()。

注意,在一個請求中,不能同時使用這兩個流!也就是說,要麼你使用repsonse.getWriter(),要麼使用response.getOutputStream(),但不能同時使用這兩個流。不然會丟擲illegalStateException異常。

3、設定響應頭資訊

可以使用response物件的setHeader()方法來設定響應頭!使用該方法設定的響應頭最終會發送給客戶端瀏覽器!

(1)response.setHeader(“content-type”, “text/html;charset=utf-8”):設定content-type響應頭,該頭的作用是告訴瀏覽器響應內容為html型別,編碼為utf-8。而且同時會設定response的字元流編碼為utf-8,即response.setCharaceterEncoding(“utf-8”);

(2)response.setHeader("Refresh","5; URL=http://www.baidu.com"):5秒後自動跳轉到百度主頁。

4、設定狀態碼及其他方法

(1)response.setContentType("text/html;charset=utf-8"):等同與呼叫response.setHeader(“content-type”, “text/html;charset=utf-8”);

(2)response.setCharacterEncoding(“utf-8”):設定字元響應流的字元編碼為utf-8;

(3)response.setStatus(200):設定狀態碼;

(4)response.sendError(404, “您要查詢的資源不存在”):當傳送錯誤狀態碼時,Tomcat會跳轉到固定的錯誤頁面去,但可以顯示錯誤資訊。

*********************************************************************************************************************************************************************************

5、重定向 (*****重點*****)

5.1 什麼是重定向(兩次請求)

當你訪問http://www.sun.com時,你會發現瀏覽器位址列中的URL會變成http://www.oracle.com/us/sun/index.htm,這就是重定向了。重定向是伺服器通知瀏覽器去訪問另一個地址,即再發出另一個請求。


5.2 如何完成重定向?

答:重定向的狀態碼為302,我們首先使用response物件向瀏覽器傳送302的狀態碼,之後再設定一個Location,即給出一個可用的URL,由瀏覽器去訪問新的URL,實現重定向。

舉例:

public class AServlet extends HttpServlet {
	public void doGet(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		response.setStatus(302); 
		response.setHeader("Location", "http://www.baidu.com"); 
	}
}

上面程式碼的作用是:當訪問AServlet後,會通知瀏覽器重定向到百度主頁。客戶端瀏覽器解析到響應碼為302後,就知道伺服器讓它重定向,所以它會馬上獲取響應頭Location,然發出第二個請求。

還有一種快捷的重定向方法,即使用response.sendRedirect()方法。比如上面例子中的兩句可以使用response.sendRedirect("http://www.baidu.com")代替。

request—封裝了客戶端所有的請求資料

1、request概述

request是Servlet.service()方法的一個引數,型別為javax.servlet.http.HttpServletRequest。在客戶端發出每個請求時,伺服器都會建立一個request物件,並把請求資料封裝到request中,然後在呼叫Servlet.service()方法時傳遞給service()方法,這說明在service()方法中可以通過request物件來獲取請求資料。

如圖所示:


request的功能可以分為以下幾種:

(1)封裝了請求頭資料;

(2)封裝了請求正文資料,如果是GET請求,那麼就沒有正文;

(3)request是一個域物件,可以把它當成Map來新增獲取資料;

(4)request提供了請求轉發和請求包含功能。

2、request域方法

request是域物件!在JavaWeb中一共四個域物件,其中ServletContext就是域物件,它在整個應用中只建立一個ServletContext物件。request其中一個,request可以在一個請求中共享資料。

一個請求會建立一個request物件,如果在一個請求中經歷了多個Servlet,那麼多個Servlet就可以使用request來共享資料。現在我們還不知道如何在一個請求中經歷幾個Servlet。

下面是request的域方法:

(1) void setAttribute(String name, Object value): 用來儲存一個物件,也可以稱之為儲存一個域屬性,例如:servletContext.setAttribute(“xxx”, “XXX”),在request中儲存了一個域屬性,域屬性名稱為xxx,域屬性的值為XXX。請注意,如果多次呼叫該方法,並且使用相同的name,那麼會覆蓋上一次的值,這一特性與Map相同;

(2) Object getAttribute(String name): 用來獲取request中的資料,當前在獲取之前需要先去儲存才行,例如:String value = (String)request.getAttribute(“xxx”);,獲取名為xxx的域屬性;

(3) void removeAttribute(String name): 用來移除request中的域屬性,如果引數name指定的域屬性不存在,那麼本方法什麼都不做;

(4) Enumeration getAttributeNames(): 獲取所有域屬性的名稱;

3、request傳遞引數

最為常見的客戶端傳遞引數方式有兩種:

(1)瀏覽器位址列直接輸入:一定是GET請求;

(2)超連結:一定是GET請求;

(3)表單:可以是GET,也可以是POST,這取決與<form>的method屬性值;

GET請求和POST請求的區別:

(1)GET請求:

請求引數會在瀏覽器的位址列中顯示,所以不安全;

請求引數長度限制長度在1K之內;

GET請求沒有請求體,無法通過request.setCharacterEncoding()來設定引數的編碼;

(2)POST請求:

請求引數不會顯示瀏覽器的位址列,相對安全;

請求引數長度沒有限制;

4、請求轉發和請求包含 (*****重點*****)

無論是請求轉發還是請求包含,都表示由多個Servlet共同來處理一個請求。例如Servlet1來處理請求,然後Servlet1又轉發給Servlet2來繼續處理這個請求。

請求轉發和請求包含
RequestDispatcher rd = request.getRequestDispatcher("/MyServlet");   使用request獲取RequestDispatcher物件,方法的引數是被轉發或包含的Servlet的Servlet路徑

請求轉發:rd.forward(request,response);

請求包含:rd.include(request,response);

有時一個請求需要多個Servlet協作才能完成,所以需要在一個Servlet跳到另一個Servlet!

> 一個請求跨多個Servlet,需要使用轉發和包含。

> 請求轉發:由下一個Servlet完成響應體!當前Servlet可以設定響應頭!(留頭不留體)            即當前Servlet設定的相應頭有效,相應體無效。

> 請求包含:由兩個Servlet共同未完成響應體!(都留)                                                                     都有效。     

> 無論是請求轉發還是請求包含,都在一個請求範圍內!使用同一個request和response!

請求轉發與請求包含比較:

(1)如果在AServlet中請求轉發到BServlet,那麼在AServlet中就不允許再輸出響應體,即不能再使用response.getWriter()和response.getOutputStream()向客戶端輸出,這一工作應該由BServlet來完成;如果是使用請求包含,那麼沒有這個限制;

(2)請求轉發雖然不能輸出響應體,但還是可以設定響應頭的,例如:response.setContentType(”text/html;charset=utf-8”);

(3)請求包含大多是應用在JSP頁面中,完成多頁面的合併;

(4)請求轉發大多是應用在Servlet中,轉發目標大多是JSP頁面;

如圖所示:


請求轉發與重定向比較

(1)請求轉發是一個請求,而重定向是兩個請求;

(2)請求轉發後瀏覽器位址列不會有變化,而重定向會有變化,因為重定向是兩個請求;

(3)請求轉發的目標只能是本應用中的資源,重定向的目標可以是其他應用;

(4)請求轉發對AServlet和BServlet的請求方法是相同的,即要麼都是GET,要麼都是POST,因為請求轉發是一個請求;

(5)重定向的第二個請求一定是GET;