jsp內置對象 轉發與重定向的區別

来源:https://www.cnblogs.com/wwkcherry/archive/2018/08/22/9520831.html
-Advertisement-
Play Games

jsp 內置對象 轉發與重定向的比較 重定向和轉發有一個重要的不同:當使用轉發時,JSP容器將使用一個內部的方法來調用目標頁面,新的頁面繼續處理同一個請求,而瀏覽器將不會知道這個過程。 與之相反,重定向方式的含義是第一個頁面通知瀏覽器發送一個新的頁面請求。因為,當你使用重定向時,瀏覽器中所顯示的UR ...


jsp 內置對象  轉發與重定向的比較

        重定向和轉發有一個重要的不同:當使用轉發時,JSP容器將使用一個內部的方法來調用目標頁面,新的頁面繼續處理同一個請求,而瀏覽器將不會知道這個過程。 與之相反,重定向方式的含義是第一個頁面通知瀏覽器發送一個新的頁面請求。因為,當你使用重定向時,瀏覽器中所顯示的URL會變成新頁面的URL, 而當使用轉發時,該URL會保持不變。重定向的速度比轉發慢,因為瀏覽器還得發出一個新的請求。同時,由於重定向方式產生了一個新的請求,所以經過一次重 定向後,request內的對象將無法使用。 
  怎麼選擇是重定向還是轉發呢?通常情況下轉發更快,而且能保持request內的對象,所以它是第一選擇。但是由於在轉發之後,瀏覽器中URL仍然指向開始頁面,此時如果重載當前頁面,開始頁面將會被重新調用。如果你不想看到這樣的情況,則選擇重定向。 

轉發和重定向的區別 :

1、重定向時瀏覽器上的網址改變,轉發是瀏覽器上的網址不變;
2、重定向實際上產生了兩次請求,轉發只有一次請求;
重定向:
  發送請求 -->伺服器運行-->響應請求,返回給瀏覽器一個新的地址與響應碼-->瀏覽器根據響應碼,判定該響應為重定向,自動發送一個新的請求給伺服器,請求地址為之前返回的地址-->伺服器運行-->響應請求給瀏覽器;
轉發:
  發送請求 -->伺服器運行-->進行請求的重新設置,例如通過request.setAttribute(name,value)-->根據轉發的地址,獲取該地址的網頁-->響應請求給瀏覽器
3、重定向時的網址可以是任何網址,轉發的網址必須是本站點的網址。
重定向:以前的request中存放的變數全部失效,併進入一個新的request作用域。
轉發:以前的request中存放的變數不會失效,就像把兩個頁面拼到了一起。
詳細介紹:
  首先是看上去不同,他們的調用分別如下:
  1、request.getRequestDispatcher("apage.jsp").forward(request, response);//轉發到apage.jsp
     2、response.sendRedirect("apage.jsp");//重定向到apage.jsp
  在jsp頁面中也允許通過下麵的方式實現轉發:
<jsp:forward page="apage.jsp" />
  重定向:以前的request中存放的變數全部失效,併進入一個新的request作用域。 
  轉發:以前的request中存放的變數不會失效,就像把兩個頁面拼到了一起。

  提 到轉發和重定向就不得不提到request作用域。很多初學者都知道當我們提交一個表單時,就創建了一個新的請求。實際上,當我們點擊一個鏈接時,也創建 了一個新的請求。那麼一個請求的作用於到底有多大呢?例如:在頁面a.jsp中有一個鏈接<a href="b.jsp?id=1"></a>這是指向b的一個鏈接,而且還帶了一個參數</a>。當我們點擊這個鏈接的時候,就產生了一個請 求,為了明確起見,我們把它叫做requestA->B。現在,在b.jsp頁面中我們就可以從這個請求中獲取信息了。在b.jsp中你可以寫入 out.println(request.getParameter("id"))進行測試。下麵更複雜一點,我們在b.jsp頁面中增加下麵的語句:
  request.setAttribute("name","funcreal"); //為請求設置一個name屬性
  out.println(request.getAttriblute("name"));//在jsp顯示name變數的值
  在b.jsp中再增加一個鏈接:<a href="c.jsp?age=23"></a>這是指向c的一個鏈接,而且還帶了一個參數</a>,當我們點擊這個連接的時候,將產生一個新的請求,這時requestA-B請求失效了,新的請求叫做requestB-C。同樣的道理,在c.jsp中,我們可以訪問到的變數只有age,因為 id,name這兩個變數都屬於requestA-B,此時它已經不存在了。下麵是源代碼:
a.jsp
<%@ page contentType="text/html; charset=GBK" %>
<html>
  <body bgcolor="#ffffff">
    <a href="b.jsp?id=1">指向b.jsp,而且還帶了一個參數id=1。requestA-B現在誕生了</a>
  </body>
</html>

b.jsp

<%@ page contentType="text/html; charset=GBK" %>
<html>
  <body bgcolor="#ffffff">
    <%
      out.println("id=" + request.getParameter("id"));
      request.setAttribute("name","Func Real");
      out.println("name=" + request.getAttribute("name"));
    %>
    <a href="c.jsp?age=23">requestA-B已經。指向c.jsp,而且還帶了一個參數age=23</a>
  </body>
</html>

c.jsp
<%@ page contentType="text/html; charset=GBK" %>
<html>
  <body bgcolor="#ffffff">
    <%
      out.println("id=" + request.getParameter("id"));
      out.println("name=" + request.getAttribute("name"));
      out.println("age=" + request.getParameter("age"));
    %>
  </body>
</html>

  那麼轉發又是怎麼回事呢?再增加一個頁面叫做d.jsp,並且在c.jsp中</body>前面增加一句<jsp:forward page="d.jsp"/>
d.jsp
<%@ page contentType="text/html; charset=GBK" %>
<html>
  <body bgcolor="#ffffff">
    requestB-C的魔爪已經伸到了d.jsp頁面
    <%
      out.println("age=" + request.getParameter("age"));
    %>
  </body>
</html>
  運行程式,發現c頁面中的內容沒有顯示出來,因為forward是自動執行的,地址欄中雖然是c.jsp但實際上,但瀏覽器中顯示的已經是d.jsp的 內容了,而且看到了從b.jsp傳過來的參數。你可以簡單得這樣理解:轉發,就是延長了requestB-C的作用 域,<jsp:forwardpage="d.jsp"/>,這一句話實際上是把c.jsp和d.jsp粘到了一起,他們就像是在一個頁面 中。

HttpServletResponse.sendRedirect 方法實現的請求重定向與RequestDispatcher.forward 方法實現的請求轉發的總結:
  1、RequestDispatcher.forward 方法只能將請求轉發給同一個WEB應用中的組件;而HttpServletResponse.sendRedirect 方法不僅可以重定向到當前應用程式中的其他資源,還可以重定向到同一個站點上的其他應用程式中的資源,甚至是使用絕對URL重定向到其他站點的資源。如果 傳遞給HttpServletResponse.sendRedirect 方法的相對URL以“/”開頭,它是相對於整個WEB站點的根目錄;如果創建RequestDispatcher 對象時指定的相對URL以“/”開頭,它是相對於當前WEB應用程式的根目錄。
  2、調用HttpServletResponse.sendRedirect 方法重定向的訪問過程結束後,瀏覽器地址欄中顯示的URL會發生改變,由初始的URL地址變成重定向的目標URL;而調用 RequestDispatcher.forward 方法的請求轉發過程結束後,瀏覽器地址欄保持初始的URL地址不變。
  3、HttpServletResponse.sendRedirect 方法對瀏覽器的請求直接作出響應,響應的結果就是告訴瀏覽器去重新發出對另外一個URL的訪問請求。
舉個例子:重定向過程好比有個綽號叫“瀏覽器”的人寫信找張三借錢,張三回信說沒有錢,讓“瀏覽器”去找李四借,並將李四現在的通信地址告訴給了“瀏覽器 ”。於是,“瀏覽器”又按張三提供通信地址給李四寫信借錢,李四收到信後就把錢匯給了“瀏覽器”。可見,“瀏覽器”一共發出了兩封信和收到了兩次回覆,“ 瀏覽器”也知道他借到的錢出自李四之手。 RequestDispatcher.forward 方法在伺服器端內部將請求轉發給另外一個資源,瀏覽器只知道發出了請求並得到了響應結果,並不知道在伺服器程式內部發生了轉發行為。這個過程好比綽號叫“ 瀏覽器”的人寫信找張三借錢,張三沒有錢,於是張三找李四借了一些錢,甚至還可以加上自己的一些錢,然後再將這些錢匯給了“瀏覽器”。可見,“瀏覽器”只 發出了一封信和收到了一次回覆,他只知道從張三那裡借到了錢,並不知道有一部分錢出自李四之手。
  4、RequestDispatcher.forward 方法的調用者與被調用者之間共用相同的request 對象和response 對象,它們屬於同一個訪問請求和響應過程;而HttpServletResponse.sendRedirect 方法調用者與被調用者使用各自的request 對象和response 對象,它們屬於兩個獨立的訪問請求和響應過程。
對於同一個WEB應用程式的內部資源之間的跳轉,特別是跳轉之前要對請求進行一些前期預處理,並要使用 HttpServletRequest.setAttribute 方法傳遞預處理結果,那就應該使用RequestDispatcher.forward 方法。
不同WEB應用程式之間的重定向,特別是要重定向到另外一個WEB站點上的資源的情況,都應該使HttpServletResponse.sendRedirect 方法。
  5、無論是RequestDispatcher.forward 方法,還是HttpServletResponse.sendRedirect 方法,在調用它們之前,都不能有內容已經被實際輸出到了客戶端。如果緩衝區中已經有了一些內容,這些內容將被從緩衝區中清除。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 統計所有小於非負整數 n 的質數的數量。 示例: 輸入: 10 輸出: 4 解釋: 小於 10 的質數一共有 4 個, 它們是 2, 3, 5, 7 。 class Solution: def countPrimes(self, n): """ :type n: int :rtype: int "" ...
  • 先來張圖: 一、程式計數器:指向當前線程正在執行的位元組碼指令的地址(或者說行號)。每個線程都有一個自己的計數器。 為什麼指向正在執行的位元組碼指令的地址?因為CPU會切換線程執行,當前線程可能會被掛起,被掛起的時候當前指令可能沒執行完成,待CPU重新調度到該線程時,需要知道當前線程執行到哪了。 二、虛 ...
  • 作為實際java開發經驗還不到一年的我,第一次寫博客,誠惶誠恐,怕把自己的謬誤公之於眾,誤人子弟,不過轉念一想,若是能有同行加以指點評判,將他們的真知灼見描述出來,那這篇文章就算拋轉引玉了。 最近在閱讀《Java編程思想》有關介面的最後一小節中,作者提出了工廠設計模式,用來解決 生成實現某個介面的對 ...
  • 才開通博客沒幾天,不知道寫什麼東西,就把最近在項目中做的東西分享一下,開通博客也可以監督一下自己,技術的路很遙遠,希望能通過寫博客來讓自己走的更遠。 一般在web項目中都需要定義一個全局異常來處理一些業務方面的異常,下麵是自定義異常的一些代碼。 1.自定義一個異常,然後繼承RuntimeExcept ...
  • 線性表的鏈式存儲 鏈式結構存儲密度小,存儲空間利用率低 只能順序存儲(其中指針域用來表明節點間的關係) 插入和刪除操作方便 代碼如下: 1 #include <stdio.h> 2 #include <stdlib.h> 3 typedef int ElemType; 4 5 typedef str ...
  • 在MyBatis中MapperProxyFactory,MapperProxy,MapperMethod是三個很重要的類。 弄懂了這3個類你就大概清楚Mapper介面與SQL的映射, 為什麼是介面,沒有實例類也可以完成註入或者調用。 其中MapperMethod可以參考:MapperMethod源碼 ...
  • 用到的: import uuid uuid是128位的全局唯一標識符, 通常用32位的一個字元串的形式來表現 uuid.uuid1() 基於MAC地址,時間戳,隨機數來生成唯一的uuid,可以保證全球範圍內的唯一性 uuid.uuid3() 通過計算一個命名空間和名字的md5散列值來給出一個uuid ...
  • 前言 在 "上一篇" 中我們學習了創建型模式的建造者模式和原型模式。本篇則來學習下結構型模式的適配器模式和橋接模式。 適配器模式 簡介 適配器模式是作為兩個不相容的介面之間的橋梁。這種類型的設計模式屬於結構型模式,它結合了兩個獨立介面的功能。 簡單的來說就是通過某個介面將不相容的兩個類進行相容,俗稱 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...