先來看一個流程圖:
服務器處理請求的流程:
(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,實現重定向。
舉例:
1
2
3
4
5
6
7
|
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;
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。
原文鏈接:http://blog.csdn.net/qq_25827845/article/details/52214614