請求響應流程圖
1】response
1 response概述
response是Servlet.service方法的一個參數,類型為javax.servlet.http.HttpServletResponse。在客戶端發出每個請求時,服務器都會創建一個response對象,並傳入給Servlet.service()方法。response對象是用來對客戶端進行響應的,這說明在service()方法中使用response對象可以完成對客戶端的響應工作。
response對象的功能分為以下四種:
- l 設置響應頭信息;
- l 發送狀態碼;
- l 設置響應正文;
- l 重定向;
2 response響應正文
response是響應對象,向客戶端輸出響應正文(響應體)可以使用response的響應流,repsonse一共提供了兩個響應流對象:
- l PrintWriter out = response.getWriter():獲取字符流;
- l ServletOutputStream out = response.getOutputStream():獲取字節流;
當然,如果響應正文內容為字符,那么使用response.getWriter(),如果響應內容是字節,例如下載時,那么可以使用response.getOutputStream()。
注意,在一個請求中,不能同時使用這兩個流!也就是說,要么你使用repsonse.getWriter(),要么使用response.getOutputStream(),但不能同時使用這兩個流。不然會拋出IllegalStateException異常。
2.1 字符響應流
字符編碼
在使用response.getWriter()時需要注意默認字符編碼為ISO-8859-1,如果希望設置字符流的字符編碼為utf-8,可以使用response.setCharaceterEncoding(“utf-8”)來設置。這樣可以保證輸出給客戶端的字符都是使用UTF-8編碼的!
但客戶端瀏覽器並不知道響應數據是什么編碼的!如果希望通知客戶端使用UTF-8來解讀響應數據,那么還是使用response.setContentType("text/html;charset=utf-8")方法比較好,因為這個方法不只會調用response.setCharaceterEncoding(“utf-8”),還會設置content-type響應頭,客戶端瀏覽器會使用content-type頭來解讀響應數據。
緩沖區
response.getWriter()是PrintWriter類型,所以它有緩沖區,緩沖區的默認大小為8KB。也就是說,在響應數據沒有輸出8KB之前,數據都是存放在緩沖區中,而不會立刻發送到客戶端。當Servlet執行結束后,服務器才會去刷新流,使緩沖區中的數據發送到客戶端。
如果希望響應數據馬上發送給客戶端:
Ø向流中寫入大於8KB的數據;
Ø調用response.flushBuffer()方法來手動刷新緩沖區;
3 設置響應頭信息
可以使用response對象的setHeader()方法來設置響應頭!使用該方法設置的響應頭最終會發送給客戶端瀏覽器!
response.setHeader(“content-type”, “text/html;charset=utf-8”):設置content-type響應頭,該頭的作用是告訴瀏覽器響應內容為html類型,編碼為utf-8。而且同時會設置response的字符流編碼為utf-8,即response.setCharaceterEncoding(“utf-8”);
response.setHeader("Refresh","5; URL=http://www.baidu.cn"):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.sendError(404, “您要查找的資源不存在”):當發送錯誤狀態碼時,Tomcat會跳轉到固定的錯誤頁面去,但可以顯示錯誤信息。
可以看出,重定向時,是服務器向游覽器重新發送了一個response命令,讓游覽器再次向url2發送請求,以獲取url2的資源
而請求轉發時,類似於是服務器自己向自己發了一個跳轉,然后將結果直接給游覽器,這也是問什么游覽器會不改變url地址。下面是具體總結他們的區別
一、response.sendRedirect(url)-服務器將新url發送給游覽器,游覽器再根據新url請求Request.getRequestDispatcher(url).forward(request,response);請求轉發到指定url(服務器內部跳轉url然后將結果發給游覽器)
二、response.sendRedirect(url)--客戶端跳轉Request.getRequestDispatcher(url).forward(request,response)--服務器跳轉
三、response.sendRedirect(url)--跳轉到指定url地址后,上個頁面的請求會結束,request對象會消亡,數據會消亡。Request.getRequestDispatcher(url).forward(request,response)--內部跳轉,request對象一直存在
四、response.sendRedirect(url)--網址會改變Request.getRequestDispatcher(url).forward(request,response)--以游覽器角度來看,他只是發送一個request然后收到一個response,所以url不變
五、response.sendRedirect(url)--傳參數需要在url后加參數。Request.getRequestDispatcher(url).forward(request,response)-傳參數可以操作request對象方法setAttribute(“name”,value)
六、response.sendRedirect(url)-可以定位到任意的網址Request.getRequestDispatcher(url).forward(request,response)-只能定位到服務器資源
七、response.sendRedirect(url)--可以重定向有frame的jsp文件Resquest.getRequestDispatcher(url).forward(request,response)-不可以重定向有frame的jsp文件,可以定向到有frame的html文件,只有在客戶端沒有輸出時才可以調用forward方法,如果當前頁面緩沖區不是空的,那么該句可能不起作用,由於這種提交不會改變url地址,刷新會重復提交。
5 重定向
5.1 什么是重定向
當你訪問http://www.sun.com時,你會發現瀏覽器地址欄中的URL會變成http://www.oracle.com/us/sun/index.htm,這就是重定向了。重定向是服務器通知瀏覽器去訪問另一個地址,即再發出另一個請求。
5.2 完成重定向
響應碼為200表示響應成功,而響應碼為302表示重定向。所以完成重定向的第一步就是設置響應碼為302。
因為重定向是通知瀏覽器再第二個請求,所以瀏覽器需要知道第二個請求的URL,所以完成重定向的第二步是設置Location頭,指定第二個請求的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.cn"); } }
上面代碼的作用是:當訪問AServlet后,會通知瀏覽器重定向到傳智主頁。客戶端瀏覽器解析到響應碼為302后,就知道服務器讓它重定向,所以它會馬上獲取響應頭Location,然發出第二個請求。
5.3 便捷的重定向方式
public class AServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.sendRedirect("http://www.baidu.cn"); } }
response.sendRedirect()方法會設置響應頭為302,以設置Location響應頭。如果要重定向的URL是在同一個服務器內,那么可以使用相對路徑,例如:
public class AServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.sendRedirect("/hello/BServlet"); } }
重定向的URL地址為:http://localhost:8080/hello/BServlet。
5.4 重定向小結
- 重定向是兩次請求;
- 重定向的URL可以是其他應用,不局限於當前應用;
- 重定向的響應頭為302,並且必須要有Location響應頭;
- 重定向就不要再使用response.getWriter()或response.getOutputStream()輸出數據,不然可能會出現異常;
2】request
1 request概述
request是Servlet.service()方法的一個參數,類型為javax.servlet.http.HttpServletRequest。在客戶端發出每個請求時,服務器都會創建一個request對象,並把請求數據封裝到request中,然后在調用Servlet.service()方法時傳遞給service()方法,這說明在service()方法中可以通過request對象來獲取請求數據。
request的功能可以分為以下幾種:
- 封裝了請求頭數據;
- 封裝了請求正文數據,如果是GET請求,那么就沒有正文;
- request是一個域對象,可以把它當成Map來添加獲取數據;
- request提供了請求轉發和請求包含功能。
2 request域方法
request是域對象!在JavaWeb中一共四個域對象,其中ServletContext就是域對象,它在整個應用中只創建一個ServletContext對象。request其中一個,request可以在一個請求中共享數據。
一個請求會創建一個request對象,如果在一個請求中經歷了多個Servlet,那么多個Servlet就可以使用request來共享數據。現在我們還不知道如何在一個請求中經歷之個Servlet,后面在學習請求轉發和請求包含后就知道了。
下面是request的域方法:
- void setAttribute(String name, Object value):用來存儲一個對象,也可以稱之為存儲一個域屬性,例如:servletContext.setAttribute(“xxx”, “XXX”),在request中保存了一個域屬性,域屬性名稱為xxx,域屬性的值為XXX。請注意,如果多次調用該方法,並且使用相同的name,那么會覆蓋上一次的值,這一特性與Map相同;
- Object getAttribute(String name):用來獲取request中的數據,當前在獲取之前需要先去存儲才行,例如:String value = (String)request.getAttribute(“xxx”);,獲取名為xxx的域屬性;
- void removeAttribute(String name):用來移除request中的域屬性,如果參數name指定的域屬性不存在,那么本方法什么都不做;
- Enumeration getAttributeNames():獲取所有域屬性的名稱;
3 request獲取請求頭數據
request與請求頭相關的方法有:
- String getHeader(String name):獲取指定名稱的請求頭;
- Enumeration getHeaderNames():獲取所有請求頭名稱;
- int getIntHeader(String name):獲取值為int類型的請求頭。
4 request獲取請求相關的其它方法
request中還提供了與請求相關的其他方法,有些方法是為了我們更加便捷的方法請求頭數據而設計,有些是與請求URL相關的方法。
int getContentLength():獲取請求體的字節數,GET請求沒有請求體,沒有請求體返回-1;
String getContentType():獲取請求類型,如果請求是GET,那么這個方法返回null;如果是POST請求,那么默認為application/x-www-form-urlencoded,表示請求體內容使用了URL編碼;
String getMethod():返回請求方法,例如:GET
Locale getLocale():返回當前客戶端瀏覽器的Locale。java.util.Locale表示國家和言語,這個東西在國際化中很有用;
String getCharacterEncoding():獲取請求編碼,如果沒有setCharacterEncoding(),那么返回null,表示使用ISO-8859-1編碼;
void setCharacterEncoding(String code):設置請求編碼,只對請求體有效!注意,對於GET而言,沒有請求體!!!所以此方法只能對POST請求中的參數有效!
String getContextPath():返回上下文路徑,例如:/hello
String getQueryString():返回請求URL中的參數,例如:name=zhangSan
String getRequestURI():返回請求URI路徑,例如:/hello/oneServlet
StringBuffer getRequestURL():返回請求URL路徑,例如:http://localhost/hello/oneServlet,即返回除了參數以外的路徑信息;
String getServletPath():返回Servlet路徑,例如:/oneServlet
String getRemoteAddr():返回當前客戶端的IP地址;
String getRemoteHost():返回當前客戶端的主機名,但這個方法的實現還是獲取IP地址;
String getScheme():返回請求協議,例如:http;
String getServerName():返回主機名,例如:localhost
int getServerPort():返回服務器端口號,例如:8080
4.1 案例:request.getRemoteAddr():封IP
可以使用request.getRemoteAddr()方法獲取客戶端的IP地址,然后判斷IP是否為禁用IP。
String ip = request.getRemoteAddr(); System.out.println(ip); if(ip.equals("127.0.0.1")) { response. getWriter().print("您的IP已被禁止!"); } else { response.getWriter().print("Hello!"); }
5 request獲取請求參數
最為常見的客戶端傳遞參數方式有兩種:
- 瀏覽器地址欄直接輸入:一定是GET請求;
- 超鏈接:一定是GET請求;
- 表單:可以是GET,也可以是POST,這取決與<form>的method屬性值;
GET請求和POST請求的區別:
GET請求:
- Ø請求參數會在瀏覽器的地址欄中顯示,所以不安全;
- Ø請求參數長度限制長度在1K之內;
- ØGET請求沒有請求體,無法通過request.setCharacterEncoding()來設置參數的編碼;
POST請求:
- Ø請求參數不會顯示瀏覽器的地址欄,相對安全;
- Ø請求參數長度沒有限制;
<a href="/hello/ParamServlet?p1=v1&p2=v2">超鏈接</a> <hr/> <form action="/hello/ParamServlet"method="post"> 參數1:<input type="text" name="p1"/><br/> 參數2:<input type="text" name="p2"/><br/> <input type="submit"value="提交"/> </form>
下面是使用request獲取請求參數的API:
- String getParameter(String name):通過指定名稱獲取參數值;
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String v1 = request.getParameter("p1"); String v2 = request.getParameter("p2"); System.out.println("p1=" + v1); System.out.println("p2=" + v2); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String v1 = request.getParameter("p1"); String v2 = request.getParameter("p2"); System.out.println("p1=" + v1); System.out.println("p2=" + v2); }
- String[] getParameterValues(String name):當多個參數名稱相同時,可以使用方法來獲取;
<a href="/hello/ParamServlet?name=zhangSan&name=liSi">超鏈接</a>
public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { String[] names = request.getParameterValues("name"); System.out.println(Arrays.toString(names)); }
- Enumeration getParameterNames():獲取所有參數的名字;
<form action="/hello/ParamServlet"method="post"> 參數1:<input type="text" name="p1"/><br/> 參數2:<input type="text" name="p2"/><br/> <input type="submit"value="提交"/> </form>
public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { Enumeration names = request.getParameterNames(); while(names.hasMoreElements()) { System.out.println(names.nextElement()); } }
- Map getParameterMap():獲取所有參數封裝到Map中,其中key為參數名,value為參數值,因為一個參數名稱可能有多個值,所以參數值是String[],而不是String。
<a href="/day05_1/ParamServlet?p1=v1&p1=vv1&p2=v2&p2=vv2">超鏈接</a>
Map<String,String[]> paramMap = request.getParameterMap(); for(String name : paramMap.keySet()) { String[] values = paramMap.get(name); System.out.println(name + ": " + Arrays.toString(values)); }
p2: [v2, vv2]
p1: [v1, vv1]
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來完成;如果是使用請求包含,那么沒有這個限制;
- 請求轉發雖然不能輸出響應體,但還是可以設置響應頭的,例如:response.setContentType(”text/html;charset=utf-8”);
- 請求包含大多是應用在JSP頁面中,完成多頁面的合並;
- 請求請求大多是應用在Servlet中,轉發目標大多是JSP頁面;
6.4 請求轉發與重定向比較
- 請求轉發是一個請求,而重定向是兩個請求;
- 請求轉發后瀏覽器地址欄不會有變化,而重定向會有變化,因為重定向是兩個請求;
- 請求轉發的目標只能是本應用中的資源,重定向的目標可以是其他應用;
- 請求轉發對AServlet和BServlet的請求方法是相同的,即要么都是GET,要么都是POST,因為請求轉發是一個請求;
- 重定向的第二個請求一定是GET;