在javaee中:轉發和重定向的區別?
第一點:
也就是最直觀的一點就是:重定向會調用HttpServletResponse.sendRedirect方法重定向的訪問過程結束后,瀏覽器地址欄中顯示的URL會發生改變.由初始的URL地址欄變成重定向的目標URL
而請求轉發過程結束后,瀏覽器的地址欄保持初始的URL地址不變.
第二點:
也就是請求發送到的位置:
HttpServletResponse.sendRedirect方法不僅可以重定向到當前應用程序中的其他資源,還可以重定向到同一個站點上的其他應用程序中的資源,甚至是使用絕對URL重定向到其他站點的資源.
而RequestDispatcher.forward方法只能將請求轉發給同一個WEB應用中的組件.
HttpServerletResponse.sendRedirect方法的相對URL以"/"開頭,它是相對於整個WEB站點的根目錄;
ReqestDispatcher對象時指定的相對URL以"/"開頭,它是相對於當前的WEB應用程序的根目錄.
第三點:重定向時,瀏覽器會收到兩次請求,
HttpServletResponse.sendRedirect方法對瀏覽器的請求直接做出響應,響應的結果就是告訴瀏覽器去從新發出另一個URL的訪問請求,
這個過程有點類似於:一個叫做"瀏覽器"的人找張三借錢,張三沒有錢,然后給"瀏覽器"說你去找北京的李四借錢.然后"瀏覽器"又跑到了北京找李四借錢,李四聽說以后,直接把錢借給了"瀏覽器"
內部轉發.瀏覽器只會收到一次請求
RequestDispatcher.forward方法在服務器端內部將請求轉發到另外一個資源.瀏覽器只知道發出了請求並得到了響應的結果.並不知道服務器內部發生了轉發行為.
這個過程就像是一個叫做"瀏覽器"的人,找張三借錢,張三自己找到李四,問李四結了錢,並把這筆錢給了"瀏覽器",而"瀏覽器"只知道這個錢是從張三手里借的.
第四點:請求時使用的request對象和response對象的不同
HttpServletResponse.sendRedirect方法調用者和被調用者使用各自的request對象和response對象.他們屬於兩個獨立的訪問請求和響應的過程.
不同的web應用程序之間的重定向,特別是要重定向到另外一個web站點上的資源的情況,都應該使用HttpServletResponse.sendRedirect方法.
RequestDispatcher.forward方法的調用者與被調用者之間共享相同的request對象和response對象,他們屬於同一個訪問請求和響應的過程.
對於同一個WEB應用程序的內部資源之間的跳轉,特別是跳轉之前要對請求進行一些前期預處理,並要使用HttpServletRequest.setAttribute方法傳遞預處理結果,那就應該使用RequestDispatcher.forward方法.
第五點:無論是RequestDispatcher.forward方法,還是HttpServletResponse.sendRedirect方法,在調用他們之前,都不能有內容已經被實際輸出到客戶端.如果緩沖區中有了一些內容,這些內容將被從緩沖區中//TODO
轉發和重定向的圖解?
兩種跳轉獲得對象的方式
//獲得轉發對象getRequestDispatcher() HttpServletRequest(httpServletRequest).getRequestDispatcher ServletContext.getRequestDispatcher(); //獲得重定向對象sendRedirect() HttpServletResponse(httpServletResponse).sendRedirect();
轉發和跳轉的小結
1.重定向時候,地址欄會發生改變也就是客戶端行為.,轉發屬於服務器內部行為,地址欄不會發生改變
2.重定向相當於是兩次請求,轉發的話是一次請求
3,重定向可以訪問不同WEB目錄下的資源,轉發只能訪問同一web路徑下的資源
4.重定向的request和response是兩個不同的,轉發的話,是同一個request和response
5重定向不可以訪問web-inf中的受保護的資源,轉發可以訪問.
6.轉發2次跳轉之間傳輸的信息不會丟失,重定向2次跳轉之間傳輸的信息會丟失(request范圍)。
轉發和重定向的選擇
1、重定向的速度比轉發慢,因為瀏覽器還得發出一個新的請求,如果在使用轉發和重定向都無所謂的時候建議使用轉發。
2、因為轉發只能訪問當前WEB的應用程序,所以不同WEB應用程序之間的訪問,特別是要訪問到另外一個WEB站點上的資源的情況,這個時候就只能使用重定向了。
轉發和重定向的應用場景
在上面我已經提到了,轉發是要比重定向快,因為重定向需要經過客戶端,而轉發沒有。有時候,采用重定向會更好,若需要重定向到另外一個外部網站,則無法使用轉發。另外,重定向還有一個應用場景:避免在用戶重新加載頁面時兩次調用相同的動作。
例如,當提交產品表單的時候,執行保存的方法將會被調用,並執行相應的動作;這在一個真實的應用程序中,很有可能將表單中的所有產品信息加入到數據庫中。但是如果在提交表單后,重新加載頁面,執行保存的方法就很有可能再次被調用。同樣的產品信息就將可能再次被添加,為了避免這種情況,提交表單后,你可以將用戶重定向到一個不同的頁面,這樣的話,這個網頁任意重新加載都沒有副作用;
但是,使用重定向不太方便的地方是,使用它無法將值輕松地傳遞給目標頁面。而采用轉發,則可以簡單地將屬性添加到Model,使得目標視圖可以輕松訪問。由於重定向經過客戶端,所以Model中的一切都會在重定向時丟失。但幸運的是,在Spring3.1版本以后,我們可以通過Flash屬性,解決重定向時傳值丟失的問題。
要使用Flash屬性,必須在Spring MVC的配置文件中添加一個<annotation-driven/>。然后,還必須再方法上添加一個新的參數類型:org.springframework.web.servlet.mvc.support.RedirectAttributes。
如下所示:
@RequestMapping(value="saveProduct",method=RequestMethod.POST) public String saveProduct(ProductForm productForm,RedirectAttributes redirectAttributes){ //執行產品保存的業務邏輯等 //傳遞參數 redirectAttributes.addFlashAttribute("message","The product is saved successfully"); //執行重定向 return "redirect:/……";
}