一,、
在一個設(shè)計良好的Web應用中,,都會綜合使用Servlet和JSP技術(shù)。Servlet控制業(yè)務流轉(zhuǎn),,JSP則負責業(yè)務處理結(jié)果的顯示,。此時,將大量用到重定向技術(shù),。 重定向技術(shù)可以分為兩類,,一類是客戶端重定向,一類是服務器端重定向,??蛻舳酥囟ㄏ蚩梢酝ㄟ^設(shè)置特定的HTTP頭,或者寫JavaScript腳本實現(xiàn)。本文主要探討服務器端重定向技術(shù)的實現(xiàn),。 服務器端的重定向相關(guān)類 服務器端的重定向技術(shù)涉及到j(luò)avax.servlet.ServletContext,、javax.servlet.RequestDispatcher、javax.servlet.http.ServletRequest,、javax.servlet.http.ServletResponse等幾個接口,。 服務器端的重定向方式 服務器端的重定向可以有兩種方式,一是使用HttpServletResponse的sendRedirect()方法,,一是使用RequestDispatcher的forward()方法,。下面對這兩種方式進行介紹。 HttpServletResponse.sendRedirect()方法 HttpServletResponse接口定義了可用于轉(zhuǎn)向的sendRedirect()方法,。代碼如下: public void sendRedirect(java.lang.String location)throws java.io.IOException 這個方法將響應定向到參數(shù)location指定的,、新的URL,。location可以是一個絕對的URL,,如response.sendRedirect("http://java.")也可以使用相對的URL。如果location以“/”開頭,,則容器認為相對于當前Web應用的根,,否則,容器將解析為相對于當前請求的URL,。這種重定向的方法,,將導致客戶端瀏覽器的請求URL跳轉(zhuǎn)。從瀏覽器中的地址欄中可以看到新的URL地址,,作用類似于上面設(shè)置HTTP響應頭信息的實現(xiàn),。 RequestDispatcher.forward()方法 RequestDispatcher是一個Web資源的包裝器,可以用來把當前request傳遞到該資源,,或者把新的資源包括到當前響應中,。RequestDispatcher接口中定義了兩個方法,參見如下代碼: public interface RequestDispatcher { void forward(ServletRequest request, ServletResponse response); void include(ServletRequest request, ServletResponse response); } forward()方法將當前的request和response重定向到該RequestDispacher指定的資源,。這在實際項目中大量使用,,因為完成一個業(yè)務操作往往需要跨越多個步驟,每一步驟完成相應的處理后,,轉(zhuǎn)向到下一個步驟,。比如,通常業(yè)務處理在Servlet中處理,,處理的結(jié)果轉(zhuǎn)向到一個JSP頁面進行顯示,。這樣看起來類似于Servlet鏈的功能,但是還有一些區(qū)別,。一個RequestDispatcher對象可以把請求發(fā)送到任意一個服務器資源,,而不僅僅是另外一個Servlet。 include()方法將把Request Dispatcher資源的輸出包含到當前輸出中。 注意,,只有在尚未向客戶端輸出響應時才可以調(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)向到外部上下文的一個服務器資源鏈接,。 3.使用javax.servlet. ServletContext的getNamedDispatcher(String name)得到名為name的一個Web資源,包括Servlet和JSP頁面,。這個資源的名字在Web應用部署描述文件web.xml中指定,。 這三種方法的使用有細微的差別。比如,,下面是一個應用的配置文件web.xml: 其中定義了兩個Servlet,,名字分別為FirstServlet和SecondServlet,對應的類分別為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)到一個其它服務器上的資源,則必須使用HttpServletResponse.sendRequest()方法,。 二,、 1.RequestDispatcher.forward() 是在服務器端起作用,當使用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之后,應該緊跟一句return; 我們已經(jīng)知道response.sendRedirect是通過瀏覽器來做轉(zhuǎn)向的,,所以只有在頁面處理完成后,,才會有實際的動作。既然你已經(jīng)要做轉(zhuǎn)向了,,那么后的輸出還有什么意義呢,?而且有可能會因為后面的輸出導致轉(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)到一個其它服務器上的資源,,則必須使用HttpServletResponse.sendRequest()方法。 3.<jsp:forward page="" /> 它的底層部分是由RequestDispatcher來實現(xiàn)的,,因此它帶有RequestDispatcher.forward()方法的印記,。 如果在<jsp:forward>之前有很多輸出,前面的輸出已使緩沖區(qū)滿,將自動輸出到客戶端,那么該語句將不起作用,這一點應該特別注意。 另外要注意:它不能改變?yōu)g覽器地址,,刷新的話會導致重復提交 |
|