1. 程式人生 > >javaweb之響應基礎

javaweb之響應基礎

請求響應流程

  • 建立request物件,請求把資料封裝到request中
  • 建立response物件
  • 呼叫servlet的service方法傳遞這兩個引數
  • 在service方法中使用request獲取請求資料,使用response完成響應

response

1.response概述

response是service的一個引數,在客戶每發出一個請求時,伺服器都會建立一個response物件,並傳入給servlet.service()方法。response物件用來對客戶端進行響應。
response分為功能四種:

  • 設定響應頭資訊
  • 傳送狀態碼
  • 設定響應正文
  • 重定向
2.response響應正文

response是響應物件,向客戶端輸出響應正文可以使用response響應流,response提供了兩個響應流物件。

  • PrintWriter out = response.getWriter();//獲取字元流
  • ServletOutputStream out = response.getOutputStream();//獲取位元組流

如果響應正文內容為字元,那麼使用response.getWriter(),如果響應內容為字元,則使用response.getOutputStream();但不能同時使用這兩個流。

2.1字元響應流
  • 字元編碼
    使用response.getWriter()時需要注意預設字元編碼ISO-8859-1,如果希望設定字元流的字元編碼為utf-8;可以使用response.setCharacterEncoding(“UTF-8”)設定,這樣可以保證客戶端的字元都是使用uft-8編碼的。同時,使用response.setContentType(“text.html;charset=utf-8”)來通知客戶端使用utf-8來解讀響應資料。

  • 緩衝區
    response.getWriter()是PrintWriter型別,所以有緩衝區,大小預設為8kb,也就是說,在響應資料沒有8KB之前,資料都存放在緩衝區當中的,不會立即發給客戶端。當servlet執行結束後,伺服器才會去重新整理流,傳送資料當客戶端。如果希望服務端響應資料立即傳送給客戶端兩種方法:1.向緩衝流中寫入大於8KB的資料。2.呼叫response.flushBuffer()方法手動重新整理緩衝區。

3.設定響應頭資訊

可以使用response物件的setHeader()方法來設定響應頭。使用該方法設定的響應頭最終會發送給客戶端。
response.setHeader(“content-type”,“text/html;charset=utf-8”);設定content-type響應頭,告訴瀏覽器響應內容為html,編碼utf-8.同時設定response.setCharaceterEncoding(“utf-8”);
eg:response.setHeader(“Refresh”,“5;URL=http://www.baidu.com”);5秒後調轉到百度主頁。

4.設定狀態碼及其他方法
  • response.setContentType(“text.html;charset=utf-8”)等同於呼叫response.setHeader(“content-type”,“text/html;charset=utf-8”)
  • response.setCharacterEncoding(“UTF-8”)設定字元響應流的字元編碼為UTF-8;
  • response.setStatus(200):設定狀態碼
  • response.setError(404,“查詢資源不存在”);當傳送錯誤狀態碼時,tomacat會跳轉到固定的錯誤頁面,也可以顯示錯誤資訊。
5.重定向
5.1什麼是重定向

當你訪問http://www.baidu.com時,你會發現瀏覽器位址列中的url會變成http://www.baidu.com/index.html,這就是重定向。重定向就是伺服器通知瀏覽器去訪問另一個地址。

5.2完成重定向

1.響應碼200表示相響應成功,302表示重定向,所以完成重定向第一步是設定響應碼為302.
2.因為重定向時通知瀏覽器完成第二個請求所以瀏覽器需要知道第二個請求的url,所以完成重定向的第二步是設定Location頭,指定第二個請求url地址。

public class AServlet extends HttpServlet{
	public void doGet(HttpServlet request,HttpServlet response){
		response.setStatus(302);//表示重定向
		response.setHeader("Location","http://www.baidu.com");//請求新的url
	}
}
當伺服器訪問AServlet時,會通知瀏覽器重定向,獲取Location響應頭,發出請求百度主頁。


便捷的重定向方式
public class AServlet extends HttpServlet{
	public void doGet(HttpServlet request,HttpServlet response){
		response.sendRedirect("http://www.baidu.com");
	}
}
response.sendRedirect();方法會設定響應頭為302,以設定Location響應頭。

如果重定向的url在同一伺服器內,則可以使用相對路徑。
public class AServlet extends HttpServlet{
	public void doGet(HttpServlet request,HttpServlet response){
		response.sendRedirect("/hello/BServlet");
	}
}
重定向地址為http://localhost:8080/hello/BServlet
5.3重定向小結
  • 重定向是兩次請求
  • 重定向的url可以是其他應用,不侷限於當前應用
  • 重定向的響應頭是302,並且必須有Location響應頭
  • 重定向就不再使用response.getWriter()或者response.getOutputStream()輸出資料,否則可能出現異常。

request

1.request概述

request是service方法的一個引數,在客戶端發出每個請求時,伺服器都會建立一個request物件,並把資料封裝到request中,然後呼叫service方法,service方法可以通過request物件來獲取請求資料。request功能分為四種:

  • 封裝了請求頭資料
  • 封裝了請求正文資料,如果是get請求,那麼沒有正文
  • request是一個域物件,可以把它當作Map來新增獲取資料‘
  • request提供了請求轉發和請求包含的功能
2.request域方法

request是一個域物件,在javaweb中一共有四個域物件,其中servletContext就是域物件,它在整個應用中只建立一個ServletContext物件,request其中一個,request可以在一個請求中共享資料。一個請求只會建立一個request物件,如果在一個請求中經歷了多servlet,那麼多個servlet就可以使用request來共享資料。
request的域方法:

  • void setAttribute(String name,Object value):用來儲存一個物件,也可以稱之為儲存一個域屬性。例如:servletContext.setAttribute(“xxx”,“XXX”);在request中儲存了一個域屬性,域屬性的名稱為xxx,域屬性的值為XXX。注意,如果多次呼叫該方法,並且使用了相同的name,那麼麼會覆蓋上一次的值。
  • Object getAttribute(String name);用來獲取request的資料,當前在獲取之前先去儲存才行。例如:String value = (String)request.getAttribute(“xxx”);獲取名稱為xxx的域屬性。
  • void removeAttribute(String naem):用來移除request的域屬性,如果name引數不存在,則什麼也不幹。
  • Enumeration getAttributeNames();獲取所有域屬性的名稱。
3.request獲取請求頭資料

request請求頭資料的相關方法:

  • String getHeader(String name):獲取指定名稱的請求頭
  • Enumeration getHeaderNames():獲取所有的請求頭名稱
  • int getIntHeader(String name):獲取值為int的請求頭
4.request獲取請求相關的其他方法

request中還提供了與請求相關的其他方法,有些方法是為了我們更加便捷的方法請求頭資料而設計,有些是與請求URL相關的方法。
l int getContentLength():獲取請求體的位元組數,GET請求沒有請求體,沒有請求體返回-1;
l String getContentType():獲取請求型別,如果請求是GET,那麼這個方法返回null;如果是POST請求,那麼預設為application/x-www-form-urlencoded,表示請求體內容使用了URL編碼;
l String getMethod():返回請求方法,例如:GET
l Locale getLocale():返回當前客戶端瀏覽器的Locale。java.util.Locale表示國家和言語,這個東西在國際化中很有用;
l String getCharacterEncoding():獲取請求編碼,如果沒有setCharacterEncoding(),那麼返回null,表示使用ISO-8859-1編碼;
l void setCharacterEncoding(String code):設定請求編碼,只對請求體有效!注意,對於GET而言,沒有請求體!!!所以此方法只能對POST請求中的引數有效!
l String getContextPath():返回上下文路徑,例如:/hello
l String getQueryString():返回請求URL中的引數,例如:name=zhangSan
l String getRequestURI():返回請求URI路徑,例如:/hello/oneServlet
l StringBuffer getRequestURL():返回請求URL路徑,例如:http://localhost/hello/oneServlet,即返回除了引數以外的路徑資訊;
l String getServletPath():返回Servlet路徑,例如:/oneServlet
l String getRemoteAddr():返回當前客戶端的IP地址;
l String getRemoteHost():返回當前客戶端的主機名,但這個方法的實現還是獲取IP地址;
l String getScheme():返回請求協議,例如:http;
l String getServerName():返回主機名,例如:localhost
l int getServerPort():返回伺服器埠號,例如:8080

5.request獲取請求引數

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

  • 瀏覽器位址列直接輸入:一定是get請求
  • 超連結:一定是get請求
  • 表單:可以是get也可以是post。

get和post區別:

  • get請求
    1.請求引數會在瀏覽器位址列中顯示,相對不安全
    2.請求引數長度限制在1KB內.
    3get沒有請求體,無法通過setcharacterEncoding設定引數的編碼
  • post請求
    1:請求引數不會顯示在位址列中
    2:請求引數長度沒有限制。
6.請求轉發於請求包含

無論是請求轉發還是請求包含,都是由多個servlet共同處理一個請求,例如:servlet1來處理請求,然後servlet1又轉發給servlet2繼續處理這個請求。

6.1請求轉發
在AServlet中,把請求轉發到BServlet:
public class AServlet extends HttpServlet {  
	public void doGet(HttpServletRequest request, HttpServletResponse response)    throws ServletException, IOException {   				  	  System.out.println("AServlet");   
	RequestDispatcher rd = request.getRequestDispatcher("/BServlet");   rd.forward(request, response);  } }	
public class BServlet extends HttpServlet {  
	public void doGet(HttpServletRequest request, HttpServletResponse response)    throws ServletException, IOException {   		   System.out.println("BServlet");  } }	
	輸出
Aservlet 
BServlet	
6.2 請求包含
在AServlet中,把請求包含到BServlet:
public class AServlet extends HttpServlet { 
	 public void doGet(HttpServletRequest request, HttpServletResponse response)    throws ServletException, IOException {   		System.out.println("AServlet");  
	  RequestDispatcher rd = request.getRequestDispatcher("/BServlet");   rd.include(request, response);  } }	
public class BServlet extends HttpServlet {  
public void doGet(HttpServletRequest request, HttpServletResponse response)    throws ServletException, IOException {   System.out.println("BServlet");  } }	
輸出
Aservlet 
BServlet	
6.3請求轉發與請求包含比較
  • 如果Aservlet中請求轉發到Bservlet,那麼Aservlet中就不再允許輸出響應體,即不能再使用response.getWriter()和response.getOutputStream();向客戶端輸出。這一工作由Bservlet完成,如果是請求包含,則沒有限制。
  • 請求轉發雖然不能輸出響應體,但可以設定響應。
  • 請求包含大多用在jsp頁面,完成多頁面的合併
  • 請求轉發大多用在servlet中,轉發目標大多數jsp頁面。
6.5請求轉發與重定向比較
  • 請求轉發是一個請求,重定向是兩個請求。
  • 請求轉發後瀏覽器位址列不會發生變化,而重定向會發生變化,因為是兩個請求。
  • 請求轉發的目標只能是本應用中的資源,重定向可以是其他應用。
  • 請求轉發對Aservlet和Bservlet的請求方法是相同的,要麼都是get要麼都是post。
  • 重定向第二個請求一定是get。