一、
在一個設(shè)計良好的Web應(yīng)用中,都會綜合使用Servlet和JSP技術(shù)。Servlet控制業(yè)務(wù)流轉(zhuǎn),JSP則負責業(yè)務(wù)處理結(jié)果的顯示。此時,將大量用到重定向技術(shù)。
重定向技術(shù)可以分為兩類,一類是客戶端重定向,一類是服務(wù)器端重定向??蛻舳酥囟ㄏ蚩梢酝ㄟ^設(shè)置特定的HTTP頭,或者寫JavaScript腳本實現(xiàn)。本文主要探討服務(wù)器端重定向技術(shù)的實現(xiàn)。
服務(wù)器端的重定向相關(guān)類
服務(wù)器端的重定向技術(shù)涉及到j(luò)avax.servlet.ServletContext、javax.servlet.RequestDispatcher、javax.servlet.http.ServletRequest、javax.servlet.http.ServletResponse等幾個接口。
服務(wù)器端的重定向方式
服務(wù)器端的重定向可以有兩種方式,一是使用HttpServletResponse的sendRedirect()方法,一是使用RequestDispatcher的forward()方法。下面對這兩種方式進行介紹。
HttpServletResponse.sendRedirect()方法
HttpServletResponse接口定義了可用于轉(zhuǎn)向的sendRedirect()方法。代碼如下:
public void sendRedirect(java.lang.String location)throws java.io.IOException
這個方法將響應(yīng)定向到參數(shù)location指定的、新的URL。location可以是一個絕對的URL,如response.sendRedirect("http://java.sun.com")也可以使用相對的URL。如果location以“/”開頭,則容器認為相對于當前Web應(yīng)用的根,否則,容器將解析為相對于當前請求的URL。這種重定向的方法,將導(dǎo)致客戶端瀏覽器的請求URL跳轉(zhuǎn)。從瀏覽器中的地址欄中可以看到新的URL地址,作用類似于上面設(shè)置HTTP響應(yīng)頭信息的實現(xiàn)。
RequestDispatcher.forward()方法
RequestDispatcher是一個Web資源的包裝器,可以用來把當前request傳遞到該資源,或者把新的資源包括到當前響應(yīng)中。RequestDispatcher接口中定義了兩個方法,參見如下代碼:
public interface RequestDispatcher {
void forward(ServletRequest request, ServletResponse response);
void include(ServletRequest request, ServletResponse response);
}
forward()方法將當前的request和response重定向到該RequestDispacher指定的資源。這在實際項目中大量使用,因為完成一個業(yè)務(wù)操作往往需要跨越多個步驟,每一步驟完成相應(yīng)的處理后,轉(zhuǎn)向到下一個步驟。比如,通常業(yè)務(wù)處理在Servlet中處理,處理的結(jié)果轉(zhuǎn)向到一個JSP頁面進行顯示。這樣看起來類似于Servlet鏈的功能,但是還有一些區(qū)別。一個RequestDispatcher對象可以把請求發(fā)送到任意一個服務(wù)器資源,而不僅僅是另外一個Servlet。 include()方法將把Request Dispatcher資源的輸出包含到當前輸出中。
注意,只有在尚未向客戶端輸出響應(yīng)時才可以調(diào)用forward()方法,如果頁面緩存不為空,在重定向前將自動清除緩存。否則將拋出一個IllegalStateException異常。
如何得到RequestDispatcher
有三種方法可以得到Request Dispatcher對象。
1.javax.servlet. ServletRequest的getRequestDispatcher(String path)方法,其中path可以是相對路徑,但不能越出當前Servlet上下文。如果path以“/”開頭,則解析為相對于當前上下文的根。
2.javax.servlet. ServletContext的getRequestDispatcher(String path)方法,其中path必須以“/”開頭,路徑相對于當前的Servlet上下文??梢哉{(diào)用ServletContext的getContext(String uripath)得到另一個Servlet上下文,并可以轉(zhuǎn)向到外部上下文的一個服務(wù)器資源鏈接。
3.使用javax.servlet. ServletContext的getNamedDispatcher(String name)得到名為name的一個Web資源,包括Servlet和JSP頁面。這個資源的名字在Web應(yīng)用部署描述文件web.xml中指定。
這三種方法的使用有細微的差別。比如,下面是一個應(yīng)用的配置文件web.xml:
其中定義了兩個Servlet,名字分別為FirstServlet和SecondServlet,對應(yīng)的類分別為org.javaresearch. redirecttest.ServletOne和org. javaresearch.redirecttest.ServletTwo??梢栽跒g覽器中通過類似于下面的鏈接訪問:
http://localhost:8080/servlet/firstservlet/
使用1中方法,例如在firstservlet可以寫入下面的代碼:
RequestDispatcher rd = request.getRequestDispatcher("secondservlet");
rd.forward(request, response);
此時控制權(quán)將轉(zhuǎn)向到第二個Servlet了。
使用2中的方法,可以從Servlet Context中得到RequestDispatcher代碼如下:
RequestDispatcher rd = getServletContext().getRequest
Dispatcher("/servlet/secondservlet");
rd.forward(request, response);
使用3中的方法,從上面的web. xml配置文件可以看到定義了兩個Servlet,名字分別為FirstServlet和SecondServlet,所以可以得到命名的Dispatcher:
RequestDispatcher rd = getServletContext().getNamedDispatcher("SecondServlet");
rd.forward(request, response);
這樣也可以重定向到SecondServlet了。
如何選擇
RequestDispatcher.forward()方法和HttpServletResponse.sendRedirect()方法的區(qū)別是:前者僅是容器中控制權(quán)的轉(zhuǎn)向,在客戶端瀏覽器地址欄中不會顯示出轉(zhuǎn)向后的地址;后者則是完全的跳轉(zhuǎn),瀏覽器將會得到跳轉(zhuǎn)的地址,并重新發(fā)送請求鏈接。這樣,從瀏覽器的地址欄中可以看到跳轉(zhuǎn)后的鏈接地址。所以,前者更加高效,在前者可以滿足需要時,盡量使用Request Dispatcher.forward()方法,并且,這樣也有助于隱藏實際的鏈接。在有些情況下,比如,需要跳轉(zhuǎn)到一個其它服務(wù)器上的資源,則必須使用HttpServletResponse.sendRequest()方法。
二、
1.RequestDispatcher.forward()
是在服務(wù)器端起作用,當使用forward()時,Servlet engine傳遞HTTP請求從當前的Servlet or JSP到另外一個Servlet,JSP 或普通HTML文件,也即你的form提交至a.jsp,在a.jsp用到了forward()重定向至b.jsp,此時form提交的所有信息在b.jsp都可以獲得,參數(shù)自動傳遞.
但forward()無法重定向至有frame的jsp文件,可以重定向至有frame的html文件,同時forward()無法在后面帶參數(shù)傳遞,比如servlet?name=frank,這樣不行,可以程序內(nèi)通過response.setAttribute("name",name)來傳至下一個頁面.
重定向后瀏覽器地址欄URL不變.
例:在servlet中進行重定向
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,IOException
{
response.setContentType("text/html; charset=gb2312");
ServletContext sc = getServletContext();
RequestDispatcher rd = null;
rd = sc.getRequestDispatcher("/index.jsp"); //定向的頁面
rd.forward(request, response);
}
通常在servlet中使用,不在jsp中使用。
2.response.sendRedirect()
是在用戶的瀏覽器端工作,sendRedirect()可以帶參數(shù)傳遞,比如servlet?name=frank傳至下個頁面,同時它可以重定向至不同的主機上,sendRedirect()可以重定向有frame.的jsp文件.
重定向后在瀏覽器地址欄上會出現(xiàn)重定向頁面的URL
例:在servlet中重定向
public void doPost(HttpServletRequest request,HttpServletResponse response)
throws ServletException,IOException
{
response.setContentType("text/html; charset=gb2312");
response.sendRedirect("/index.jsp");
}
由于response是jsp頁面中的隱含對象,故在jsp頁面中可以用response.sendRedirect()直接實現(xiàn)重定位。
注意:
(1).使用response.sendRedirect時,前面不能有HTML輸出。
這并不是絕對的,不能有HTML輸出其實是指不能有HTML被送到了瀏覽器。事實上現(xiàn)在的server都有cache機制,一般在8K(我是說JSP SERVER),這就意味著,除非你關(guān)閉了cache,或者你使用了out.flush()強制刷新,那么在使用sendRedirect之前,有少量的HTML輸出也是允許的。
(2).response.sendRedirect之后,應(yīng)該緊跟一句return;
我們已經(jīng)知道response.sendRedirect是通過瀏覽器來做轉(zhuǎn)向的,所以只有在頁面處理完成后,才會有實際的動作。既然你已經(jīng)要做轉(zhuǎn)向了,那么后的輸出還有什么意義呢?而且有可能會因為后面的輸出導(dǎo)致轉(zhuǎn)向失敗。
比較:
(1).Request Dispatcher.forward()是容器中控制權(quán)的轉(zhuǎn)向,在客戶端瀏覽器地址欄中不會顯示出轉(zhuǎn)向后的地址;
(2).response.sendRedirect()則是完全的跳轉(zhuǎn),瀏覽器將會得到跳轉(zhuǎn)的地址,并重新發(fā)送請求鏈接。這樣,從瀏覽器的地址欄中可以看到跳轉(zhuǎn)后的鏈接地址。
前者更加高效,在前者可以滿足需要時,盡量使用RequestDispatcher.forward()方法.
注:在有些情況下,比如,需要跳轉(zhuǎn)到一個其它服務(wù)器上的資源,則必須使用HttpServletResponse.sendRequest()方法。
3.<jsp:forward page="" />
它的底層部分是由RequestDispatcher來實現(xiàn)的,因此它帶有RequestDispatcher.forward()方法的印記。
如果在<jsp:forward>之前有很多輸出,前面的輸出已使緩沖區(qū)滿,將自動輸出到客戶端,那么該語句將不起作用,這一點應(yīng)該特別注意。
另外要注意:它不能改變?yōu)g覽器地址,刷新的話會導(dǎo)致重復(fù)提交
本站僅提供存儲服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請
點擊舉報。