forward和redirect引出的路徑問題

轉:http://77857.blog.51cto.com/67857/167022

1.  絕對路徑與相對路徑

如果咱們使用的URL網址是以“/”開頭的,那麼這個網址就叫做絕對路徑。
如果咱們使用的URL網址不是“/”開頭的,那麼這個網址就叫做相對路徑。
2. 相對路徑
在相對路徑上,兩者的表現是相同的。
看看lingo-sample/03-03/這個例子,如果我們去請求relative/forward.jsp或redirect.jsp,然後從這裏再跳轉向它下面的result/result.jsp會怎樣呢?
forward的例子:
<%request.getRequestDispatcher("result/result.jsp").forward(request, response);%>
                        
這裏的相對路徑就是result/result.jsp。
因爲剛剛請求的test.jsp是在/03-03/relative/下,所以我們的當前路徑就是/03-03/relative/,執行forward的時候會尋找當前路徑下的result/result.jsp,找到之後便轉發請求。
redirect的例子:
<%response.sendRedirect("result/result.jsp");%>
                        
這裏的相對路徑也是result/result.jsp。
因爲剛剛請求的test.jsp是在/03-03/relative/下,所以我們的當前路徑就是/03-03/relative/,執行redirect的時候會把當前路徑加上result/result.jsp,把結果作爲重定向的地址發送給瀏覽器,瀏覽器再去請求/03-03/relative/result/result.jsp,從而得到響應。
3. 絕對路徑
問題出現了,絕對路徑在forward和redirect中出現了差別,還是剛纔的情況,但使用絕對路徑的時候寫法便不同了。
forward的例子:
<%request.getRequestDispatcher("/relative/result/result.jsp").forward(request, response);%>
                        
這裏的絕對路徑就是/relative/result/result.jsp。
在本地測試時,forward把http://localhost:8080/03-03/當作根路徑,在它的基礎上計算絕對路徑。
這是由jsp的部署方式決定的,webapp裏可以放好多項目,爲了讓這些項目可以互不影響、獨立運行,不能讓請求從一個項目直接在服務器內部轉移到另一個項目。爲了防止出現這種情況,在執行forward的時候乾脆把項目的路徑當作根目錄,開發者看不到其他項目,也就不會出現問題了。
redirect的例子:
<%response.sendRedirect("/03-03/absolute/result/result.jsp");%>
                        
這裏的絕對路徑卻是/03-03/absolute/result/result.jsp。
在本地測試時,redirect把http://localhost:8080/當作根路徑,在它的基礎上計算絕對路徑。
因爲redirect會讓瀏覽器重新發起一個新請求,所以不會攪亂服務器裏多個項目之間的關係,也就不需要對它做限制,如果需要在多個項目之間進行跳轉,就只能使用redirect。不過因爲重新發起了新的請求,上次請求的那些數據都會丟失,如果有什麼重要的數據,記得要重新設置。
4. forward導致找不到圖片
找不到圖片,找不到js腳本,找不到css樣式表,都屬於這個問題。
要演示這個問題,是非常容易的,只需要滿足兩個條件:
forward前後的jsp頁面不在一個目錄下。
forward後的jsp頁面裏使用相對路徑引用一些資源,圖片,js腳本,css樣式表什麼的。
03-04裏就模擬了這樣一個環境,你進入http://localhost:8080/03-04/,選擇“有問題的”:

打開03-04可以看到如下的目錄結構:
|--+ 03-04
   |--- index.jsp
   |--- test.jsp
   |--+ result
      |--- success.jsp
      |--- failure.jsp
      |--- lingo.png
            
剛纔咱們看到的頁面是failure.jsp,它裏邊顯示圖片的部分是:
<img src="lingo.png" />
            
這時候就有疑問了,lingo.png和failure.jsp明明在同一個目錄下,爲什麼無法顯示。
現在請在無法顯示的圖片上,點擊鼠標右鍵,選擇屬性,讓我們看一下圖片的請求地址:

圖片的位置本來在http://localhost:8080/03-04/result/lingo.png,但請求的地址卻是http://localhost:8080/03-04/lingo.png。問題就是丟掉了中間的/result。
再試一次index.jsp上的“沒問題的”:

這次我們看到的頁面是success.jsp,它裏邊顯示圖片的部分是:
<img src="result/lingo.png" />
            
結果手工加上result這段路徑後就可以顯示圖片了。
這個問題還要追溯到瀏覽器對html的處理方式,在html裏包含的圖片,css樣式表,js腳本,視頻等等外部資源,都需要瀏覽器再次向服務器發起請求。
如果這些外部資源使用了相對路徑,瀏覽器就會在當前請求路徑的基礎上,加上相對路徑拼接出完整的http請求,發送給服務器。這個例子中,我們請求http://localhost:8080/03-04/test.jsp,瀏覽器得到的當前路徑就是http://localhost:8080/03-04/,failure.jsp中圖片的相對路徑是lingo.png,那麼拼接的結果是http://localhost:8080/03-04/lingo.png。
不要怪瀏覽器太傻,是因爲使用forward的時候瀏覽器並不清楚這些改變。它一直認爲,既然自己請求的是test.jsp,返回的自然就是test.jsp的內容,那麼再使用test.jsp當作當前路徑去計算相對路徑當然沒有問題。是我們欺騙了瀏覽器,在服務器偷偷改變了請求流向,返回了其他頁面的內容。
清楚了以上的請求流程,就知道如何應對這種問題了。
第一種方法:不要在不同目錄之間使用forward做請求轉發,保證當前路徑不發生變化。
第二種方法:像上例一樣修改圖片路徑,或全部改爲絕對路徑。
請根據實際需要進行選擇。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章