JSP中forward和redirect有什麼區別? 什麼時候必須用哪個?
解釋一:
RequestDispatcher.forward()方法和HttpServletResponse.sendRedirect()方法的區別是:前者僅是容器中控制權的轉向,在客戶端瀏覽器位址列中不會顯示出轉向後的地址,他是不會改變Request的值,如果你需要在下一個頁面中能從中獲取新的資訊的話,你可以Request.setAttribute()來放置一些標誌,這樣從下一個頁面中獲取;後者則是完全的跳轉,瀏覽器將會得到跳轉的地址,並重新發送請求連結。這樣,從瀏覽器的位址列中可以看到跳轉後的連結地址。所以,前者更加高效,在前者可以滿足需要時,儘量使用Request
Dispatcher.forward()方法,並且,這樣也有助於隱藏實際的連結。在有些情況下,比如,需要跳轉到一個其它伺服器上的資源,則必須使用HttpServletResponse.sendRequest()方法。
forward是伺服器請求資源,伺服器直接訪問目標地址的URL,把那個URL的響應內容讀取過來,然後把這些內容再發給瀏覽器,瀏覽器根本不知道伺服器傳送的內容是從哪兒來的,所以它的位址列中還是原來的地址。 |
解釋三:
-
使用 HttpServletResponse.sendRedirect() 方法
-
使用 servlet 中的 RequestDispatcher.forward() 方法或 JSP 頁面中的特殊標籤:這會導致呼叫此方法的資源終止,同時終止對作為轉發目標的資源的呼叫。對客戶端瀏覽器來說,這看起來就像是單個請求。例如,如果將請求從 servlet 轉發到 JSP 頁面,則您可能要附加該 servlet 中的某些處理結果,以便這些結果可以在 JSP 頁面中顯示出來。
所以response.sendRedirect()不能用request物件傳遞資料,而forward是可以的。 (驗證屬實)
解釋四:
由於response是jsp頁面中的隱含物件,故在jsp頁面中可以用response.sendRedirect()直接實現重定位。
注意:
(1).使用response.sendRedirect時,前面不能有HTML輸出。
這並不是絕對的,不能有HTML輸出其實是指不能有HTML被送到了瀏覽器。事實上現在的server都有cache機制,一般在8K(我是說JSP SERVER),這就意味著,除非你關閉了cache,或者你使用了out.flush()強制重新整理,那麼在使用sendRedirect之前,有少量的HTML輸出也是允許的。
(2).response.sendRedirect之後,應該緊跟一句return;
我們已經知道response.sendRedirect是通過瀏覽器來做轉向的,所以只有在頁面處理完成後,才會有實際的動作。既然你已經要做轉向了,那麼後的輸出還有什麼意義呢?而且有可能會因為後面的輸出導致轉向失敗。
五補充
1.RequestDispatcher.forward()
是在伺服器端起作用,當使用forward()時,Servlet engine傳遞HTTP請求從當前的Servlet or JSP到另外一個Servlet,JSP 或普通HTML檔案,也即你的form提交至a.jsp,在a.jsp用到了forward()重定向至b.jsp,此時form提交的所有資訊在b.jsp都可以獲得,引數自動傳遞.
但forward()無法重定向至有frame的jsp檔案,可以重定向至有frame的html檔案,同時forward()無法在後面帶引數傳遞,比如servlet?name=frank,這樣不行,可以程式內通過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()可以帶引數傳遞,比如servlet?name=frank傳至下個頁面,同時它可以重定向至不同的主機上,sendRedirect()可以重定向有frame.的jsp檔案.
重定向後在瀏覽器位址列上會出現重定向頁面的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()直接實現重定位。
注意:
(1).使用response.sendRedirect時,前面不能有HTML輸出
這並不是絕對的,不能有HTML輸出其實是指不能有HTML被送到了瀏覽器。事實上現在的server都有cache機制,一般在8K(我是說JSP SERVER),這就意味著,除非你關閉了cache,或者你使用了out.flush()強制重新整理,那麼在使用sendRedirect之前,有少量的HTML輸出也是允許的。
(2).response.sendRedirect之後,應該緊跟一句return;
我們已經知道response.sendRedirect是通過瀏覽器來做轉向的,所以只有在頁面處理完成後,才會有實際的動作。既然你已經要做轉向了,那麼後的輸出還有什麼意義呢?而且有可能會因為後面的輸出導致轉向失敗。
六比較: 共同點: 都實現了根據自己的條件實現的頁面得跳轉。 (1).Request Dispatcher.forward()是容器中控制權的轉向,在客戶端瀏覽器位址列中不會顯示出轉向後的地址;
(2).response.sendRedirect()則是完全的跳轉,瀏覽器將會得到跳轉的地址,並重新發送請求連結。這樣,從瀏覽器的位址列中可以看到跳轉後的連結地址。
前者更加高效,在前者可以滿足需要時,儘量使用RequestDispatcher.forward()方法.
注:在有些情況下,比如,需要跳轉到一個其它伺服器上的資源,則必須使用HttpServletResponse.sendRequest()方法。
3.<jsp:forward page="" />
它的底層部分是由RequestDispatcher來實現的,因此它帶有RequestDispatcher.forward()方法的印記。
如果在<jsp:forward>之前有很多輸出,前面的輸出已使緩衝區滿,將自動輸出到客戶端,那麼該語句將不起作用,這一點應該特別注意。
另外要注意:它不能改變瀏覽器地址,重新整理的話會導致重複提交
4.修改HTTP header的Location屬性來重定向
通過設定直接修改位址列來實現頁面的重定向。
jsp檔案程式碼如下:
<% response.setStatus(HttpServletResponse.SC_MOVED_PERMANENTLY); String newLocn = "/newpath/jsa.jsp"; response.setHeader("Location",newLocn); %> |
5.JSP中實現在某頁面停留若干秒後,自動重定向到另一頁面
在html檔案中,下面的程式碼:
<meta http-equiv="refresh" content="300; url=target.jsp"> |
它的含義:在5分鐘之後正在瀏覽的頁面將會自動變為target.html這一頁。程式碼中300為重新整理的延遲時間,以秒為單位。targer.html為你想轉向的目標頁,若為本頁則為自動重新整理本頁。
由上可知,可以通過setHeader來實現某頁面停留若干秒後,自動重定向到另一頁面。
關鍵程式碼:
String content=stayTime+";URL="+URL; response.setHeader("REFRESH",content); |
類別 | 概念 | 共享資料 | 應用 |
Redirect | URL重新定向:可以是任意的URL | 不能共享request裡面的資料 | 一般用於使用者登出登入時返回主頁面和跳轉到其它的網站等等 |
Forward | 頁面的轉發:只能是同一個Web應用程式的其他Web元件 | 轉發頁面和轉發到的頁面可以共性request裡面的資料 | 一般用於使用者登入的時候根據角色轉發到相應的模組等等 |