2019-9-12 seo達(dá)人
轉(zhuǎn)發(fā)和重定向區(qū)別詳解
作為一名程序員,特別是java web開發(fā)的程序員,在使用servlet/jsp的時候,我們必須要知道實現(xiàn)頁面跳轉(zhuǎn)的兩種方式的區(qū)別和聯(lián)系:即轉(zhuǎn)發(fā)和重定向的區(qū)別。
1、RequestDispatcher.forward方法只能將請求轉(zhuǎn)發(fā)給同一個WEB應(yīng)用中的組件;而HttpServletResponse.sendRedirect 方法不僅可以重定向到當(dāng)前應(yīng)用程序中的其他資源,還可以重定向到同一個站點上的其他應(yīng)用程序中的資源,甚至是使用絕對URL重定向到其他站點的資源。如果傳遞給HttpServletResponse.sendRedirect 方法的相對URL以“/”開頭,它是相對于整個WEB站點的根目錄;如果創(chuàng)建RequestDispatcher對象時指定的相對URL以“/”開頭,它是相對于當(dāng)前WEB應(yīng)用程序的根目錄。、
2、調(diào)用HttpServletResponse.sendRedirect方法重定向的訪問過程結(jié)束后,瀏覽器地址欄中顯示的URL會發(fā)生改變,由初始的URL地址變成重定向的目標(biāo)URL;而調(diào)用RequestDispatcher.forward 方法的請求轉(zhuǎn)發(fā)過程結(jié)束后,瀏覽器地址欄保持初始的URL地址不變。
3、HttpServletResponse.sendRedirect方法對瀏覽器的請求直接作出響應(yīng),響應(yīng)的結(jié)果就是告訴瀏覽器去重新發(fā)出對另外一個URL的 訪問請求,這個過程好比有個綽號叫“瀏覽器”的人寫信找張三借錢,張三回信說沒有錢,讓“瀏覽器”去找李四借,并將李四現(xiàn)在的通信地址告訴給了“瀏覽器”。于是,“瀏覽器”又按張三提供通信地址給李四寫信借錢,李四收到信后就把錢匯給了“瀏覽器”??梢?,“瀏覽器”一共發(fā)出了兩封信和收到了兩次回復(fù), “瀏覽器”也知道他借到的錢出自李四之手。RequestDispatcher.forward方法在服務(wù)器端內(nèi)部將請求轉(zhuǎn)發(fā)給另外一個資源,瀏覽器只知道發(fā)出了請求并得到了響應(yīng)結(jié)果,并不知道在服務(wù)器程序內(nèi)部發(fā)生了轉(zhuǎn)發(fā)行為。這個過程好比綽號叫“瀏覽器”的人寫信找張三借錢,張三沒有錢,于是張三找李四借了一些錢,甚至還可以加上自己的一些錢,然后再將這些錢匯給了“瀏覽器”??梢姡盀g覽器”只發(fā) 出了一封信和收到了一次回復(fù),他只知道從張三那里借到了錢,并不知道有一部分錢出自李四之手。
4、RequestDispatcher.forward方法的調(diào)用者與被調(diào)用者之間共享相同的request對象和response對象,它們屬于同一個訪問請求和響應(yīng)過程;而HttpServletResponse.sendRedirect方法調(diào)用者與被調(diào)用者使用各自的request對象和response對象,它們屬于兩個獨立的訪問請求和響應(yīng)過程。對于同一個WEB應(yīng)用程序的內(nèi)部資源之間的跳轉(zhuǎn),特別是跳轉(zhuǎn)之前要對請求進(jìn)行一些前期預(yù)處理,并要使用HttpServletRequest.setAttribute方法傳遞預(yù)處理結(jié)果,那就應(yīng)該使用RequestDispatcher.forward方法。不同WEB應(yīng)用程序之間的重定向,特別是要重定向到另外一個WEB站點上的資源的情況,都應(yīng)該使用HttpServletResponse.sendRedirect方法。
5、無論是RequestDispatcher.forward方法,還是HttpServletResponse.sendRedirect方法,在調(diào)用它們之前,都不能有內(nèi)容已經(jīng)被實際輸出到了客戶端。如果緩沖區(qū)中已經(jīng)有了一些內(nèi)容,這些內(nèi)容將被從緩沖區(qū)中。
以上五點的論述來源于:點擊查看原文論述
轉(zhuǎn)發(fā)和重定向的圖解
兩種跳轉(zhuǎn)獲得對象的方式
//獲得轉(zhuǎn)發(fā)對象getRequestDispatcher()
HttpServletRequest(httpServletRequest).getRequestDispatcher
ServletContext.getRequestDispatcher();
//獲得重定向?qū)ο髎endRedirect()
HttpServletResponse(httpServletResponse).sendRedirect();
轉(zhuǎn)發(fā)和跳轉(zhuǎn)的小結(jié)
1、轉(zhuǎn)發(fā)使用的是getRequestDispatcher()方法;重定向使用的是sendRedirect();
2、轉(zhuǎn)發(fā):瀏覽器URL的地址欄不變。重定向:瀏覽器URL的地址欄改變;
3、轉(zhuǎn)發(fā)是服務(wù)器行為,重定向是客戶端行為;
4、轉(zhuǎn)發(fā)是瀏覽器只做了一次訪問請求。重定向是瀏覽器做了至少兩次的訪問請求;
5、轉(zhuǎn)發(fā)2次跳轉(zhuǎn)之間傳輸?shù)男畔⒉粫G失,重定向2次跳轉(zhuǎn)之間傳輸?shù)男畔G失(request范圍)。
轉(zhuǎn)發(fā)和重定向的選擇
1、重定向的速度比轉(zhuǎn)發(fā)慢,因為瀏覽器還得發(fā)出一個新的請求,如果在使用轉(zhuǎn)發(fā)和重定向都無所謂的時候建議使用轉(zhuǎn)發(fā)。
2、因為轉(zhuǎn)發(fā)只能訪問當(dāng)前WEB的應(yīng)用程序,所以不同WEB應(yīng)用程序之間的訪問,特別是要訪問到另外一個WEB站點上的資源的情況,這個時候就只能使用重定向了。
轉(zhuǎn)發(fā)和重定向的應(yīng)用場景
在上面我已經(jīng)提到了,轉(zhuǎn)發(fā)是要比重定向快,因為重定向需要經(jīng)過客戶端,而轉(zhuǎn)發(fā)沒有。有時候,采用重定向會更好,若需要重定向到另外一個外部網(wǎng)站,則無法使用轉(zhuǎn)發(fā)。另外,重定向還有一個應(yīng)用場景:避免在用戶重新加載頁面時兩次調(diào)用相同的動作。
例如,當(dāng)提交產(chǎn)品表單的時候,執(zhí)行保存的方法將會被調(diào)用,并執(zhí)行相應(yīng)的動作;這在一個真實的應(yīng)用程序中,很有可能將表單中的所有產(chǎn)品信息加入到數(shù)據(jù)庫中。但是如果在提交表單后,重新加載頁面,執(zhí)行保存的方法就很有可能再次被調(diào)用。同樣的產(chǎn)品信息就將可能再次被添加,為了避免這種情況,提交表單后,你可以將用戶重定向到一個不同的頁面,這樣的話,這個網(wǎng)頁任意重新加載都沒有副作用;
但是,使用重定向不太方便的地方是,使用它無法將值輕松地傳遞給目標(biāo)頁面。而采用轉(zhuǎn)發(fā),則可以簡單地將屬性添加到Model,使得目標(biāo)視圖可以輕松訪問。由于重定向經(jīng)過客戶端,所以Model中的一切都會在重定向時丟失。但幸運的是,在Spring3.1版本以后,我們可以通過Flash屬性,解決重定向時傳值丟失的問題。
要使用Flash屬性,必須在Spring MVC的配置文件中添加一個<annotation-driven/>。然后,還必須再方法上添加一個新的參數(shù)類型:org.springframework.web.servlet.mvc.support.RedirectAttributes。
如下所示:
@RequestMapping(value="saveProduct",method=RequestMethod.POST)
public String saveProduct(ProductForm productForm,RedirectAttributes redirectAttributes){
//執(zhí)行產(chǎn)品保存的業(yè)務(wù)邏輯等
//傳遞參數(shù)
redirectAttributes.addFlashAttribute("message","The product is saved successfully");
//執(zhí)行重定向
return "redirect:/……";
藍(lán)藍(lán)設(shè)計( m.820esy.cn )是一家專注而深入的界面設(shè)計公司,為期望卓越的國內(nèi)外企業(yè)提供卓越的UI界面設(shè)計、BS界面設(shè)計 、 cs界面設(shè)計 、 ipad界面設(shè)計 、 包裝設(shè)計 、 圖標(biāo)定制 、 用戶體驗 、交互設(shè)計、 網(wǎng)站建設(shè) 、平面設(shè)計服務(wù)。
藍(lán)藍(lán)設(shè)計的小編 http://m.820esy.cn