JSP的四個作用域

首先要聲明一點,所謂“作用域”就是“信息共享的範圍”,也就是說一個信息能夠在多大的範圍內有效。4個JSP內置對象的作用域分別爲:application、session、request、page 。JSP內置對象作用域表如下:

名稱

作用域

application

在所有應用程序中有效

session

在當前會話中有效

request

在當前請求中有效

page

在當前頁面有效

Web交互的最基本單位爲HTTP請求。每個用戶從進入網站到離開網站這段過程稱爲一個HTTP會話,一個服務器的運行過程中會有多個用戶訪問,就是多個HTTP會話。作用域解釋如下。

application 作用域

     如果把變量放到application裏,就說明它的作用域是application,它的有效範圍是整個應用。 整個應用是指從應用啓動,到應用結束。我們沒有說“從服務器啓動,到服務器關閉”,是因爲一個服務器可能部署多個應用,當然你關閉了服務器,就會把上面所有的應用都關閉了。 application作用域裏的變量,它們的存活時間是最長的,如果不進行手工刪除,它們就一直可以使用。

application作用域上的信息傳遞是通過ServletContext實現的,它提供的主要方法如下所示:

Object getAttribute(String name)  //從application中獲取信息;

void setAttribute(String name, Object value)  //向application作用域中設置信息。

session作用域

       session作用域比較容易理解,同一瀏覽器對服務器進行多次訪問,在這多次訪問之間傳遞信息,就是session作用域的體現。如果把變量放到session裏,就說明它的作用域是session,它的有效範圍是當前會話。所謂當前會話,就是指從用戶打開瀏覽器開始,到用戶關閉瀏覽器這中間的過程。這個過程可能包含多個請求響應。也就是說,只要用戶不關瀏覽器,服務器就有辦法知道這些請求是一個人發起的,整個過程被稱爲一個會話(session),而放到會話中的變量,就可以在當前會話的所有請求裏使用。 
 session是通過HttpSession接口實現的,它提供的主要方法如下所示:

Object HttpSession.getAttribute(String name)  //從session中獲取信息。

void HttpSession.setAttribute(String name, Object value)//向session中保存信息。

HttpSession HttpServletRequest.getSessio()  //獲取當前請求所在的session的對象。

 session的開始時刻比較容易判斷,它從瀏覽器發出第一個HTTP請求即可認爲會話開始。但結束時刻就不好判斷了,因爲瀏覽器關閉時並不會通知服務器,所以只能通過如下這種方法判斷:如果一定的時間內客戶端沒有反應,則認爲會話結束。Tomcat的默認值爲120分鐘,但這個值也可以通過HttpSession的setMaxInactiveInterval()方法來設置:

void setMaxInactiveInterval(int interval)

如果想主動讓會話結束,例如用戶單擊“註銷”按鈕的時候,可以使用 HttpSession 的 invalidate()方法,用於強制結束當前session:void invalidate()

request作用域

     一個HTTP請求的處理可能需要多個Servlet合作,而這幾個Servlet之間可以通過某種方式傳遞信息,但這個信息在請求結束後就無效了。request裏的變量可以跨越forward前後的兩頁。但是隻要刷新頁面,它們就重新計算了。如果把變量放到request裏,就說明它的作用域是request,它的有效範圍是當前請求週期。 所謂請求週期,就是指從http請求發起,到服務器處理結束,返回響應的整個過程。在這個過程中可能使用forward的方式跳轉了多個jsp頁面,在這些頁面裏你都可以使用這個變量。

Servlet之間的信息共享是通過HttpServletRequest接口的兩個方法來實現的:

void setAttribute(String name, Object value)  //將對象value以name爲名稱保存到request作用域中。

Object getAttribute(String name) //從request作用域中取得指定名字的信息。

     JSP中的doGet()、doPost()方法的第一個參數就是HttpServletRequest對象,使用這個對象的 setAttribute()方法即可傳遞信息。那麼在設置好信息之後,要通過何種方式將信息傳給其他的Servlet呢?這就要用到RequestDispatcher接口的forward()方法,通過它將請求轉發給其他Servlet

 RequestDispatcher ServletContext.getRequestDispatcher(String path) //取得Dispatcher以便轉發,path爲轉發的目的Servlet。

 void RequestDispatcher.forward(ServletRequest request, ServletResponse response)//將request和response轉發

     因此,只需要在當前Servlet中先通過setAttribute()方法設置相應的屬性,然後使用forward()方法進行跳轉,最後在跳轉到的Servlet中通過使用getAttribute()方法即可實現信息傳遞。

  需要注意兩點:

1、轉發不是重定向,轉發是在Web應用內部進行的。

2、轉發對瀏覽器是透明的,也就是說,無論在服務器上如何轉發,瀏覽器地址欄中顯示的仍然是最初那個Servlet的地址。

page作用域

page對象的作用範圍僅限於用戶請求的當前頁面,對於page對象的引用將在響應返回給客戶端之後被釋放,或者在請求被轉發到其他地方後被釋放。page裏的變量只要頁面跳轉了,它們就不見了如果把變量放到pageContext裏,就說明它的作用域是page,它的有效範圍只在當前jsp頁面裏。從把變量放到pageContext開始,到jsp頁面結束,你都可以使用這個變量。

以上介紹的作用範圍越來越小,request和page的生命週期都是短暫的,它們之間的區別:一個request可以包含多個page頁(include,forward及filter)。

爲了讓大家更容易理解application、session、request、page 4個對象的作用範圍,我們給出兩個程序來進行詳細說明。

 

【程序1】page01.jsp

[html] view plaincopy
  1. <%@ page language="java" contentType="text/html; charset=UTF-8"    
  2.     pageEncoding="UTF-8"%>    
  3. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">    
  4. <jsp:useBean id="pagevar" scope="page" class="java.lang.StringBuffer"/>    
  5. <jsp:useBean id="requestvar" scope="request" class="java.lang.StringBuffer"/>    
  6. <jsp:useBean id="sessionvar" scope="session"  class="java.lang.StringBuffer"/>    
  7.  <jsp:useBean id="applicationvar" scope="application"  class="java.lang.StringBuffer"/>    
  8. <html>    
  9. <head>    
  10. <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">    
  11. <title>JSP內置對象作用域</title>    
  12. </head>    
  13. <body>    
  14.  <%    
  15.     pagevar.append("page01");    
  16.     requestvar.append("page01");    
  17.     sessionvar.append("page01");    
  18.     applicationvar.append("page01");    
  19. %>    
  20. <jsp:forward page="page02.jsp"/>    
  21. </body>    
  22. </html>    

 

【程序2】page02.jsp

[html] view plaincopy
  1. <%@ page language="java" import="java.util.*" contentType="text/html; charset=UTF-8"    
  2.     pageEncoding="UTF-8"%>    
  3. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">    
  4. <jsp:useBean id="pagevar" scope="page" class="java.lang.StringBuffer"/>    
  5. <jsp:useBean id="requestvar" scope="request" class="java.lang.StringBuffer"/>    
  6. <jsp:useBean id="sessionvar" scope="session" class="java.lang.StringBuffer"/>    
  7. <jsp:useBean id="applicationvar" scope="application" class="java.lang.StringBuffer"/>    
  8. <html>    
  9. <head>    
  10. <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">    
  11. <title>JSP內置對象作用域</title>    
  12. </head>    
  13. <body>    
  14. <%    
  15.     pagevar.append("page02");    
  16.     requestvar.append("page02");    
  17.     sessionvar.append("page02");    
  18.     applicationvar.append("page02");    
  19. %>    
  20. page = <%=pagevar.toString()%><br/>    
  21. request = <%=requestvar.toString()%><br/>    
  22. session = <%=sessionvar.toString()%><br/>    
  23. application = <%=applicationvar.toString()%><br/>    
  24. </body>    
  25. </html>    

 

測試步驟以及結果分析:

1、直接運行程序1的結果爲:(圖1)

程序1的結果

我們看到,page的作用域的值爲page02,說明確實只在當前的頁面起作用,即跳轉到的page2頁面request的作用域在當前請求中有效所以其值爲程序1和跳轉到程序2之和;session的作用域爲當前會話,所以其值也是程序1和跳轉到程序2之和;而application對所有應用有效,也就是只要在應用,都要疊加,即程序1中的值與程序2中的值的疊加;

 

2、不要關閉程序1運行的瀏覽器,直接運行程序2,其結果爲:(圖2)

保持程序1運行的瀏覽器,直接運行程序2的結果圖

對比圖1的結果,我們發現page作用域沒有變化,它的值只是程序2裏的值;request作用域僅在當前請求作用,故也以程序2的值爲準,變成page02;session的作用域爲當前會話,因爲運行程序1的瀏覽器保持着,說明還處於同一會話中,所以要在之前的基礎上疊加上一個page02;而application對所有應用有效,也就是只要在應用,都要疊加,即在之前的基礎上疊加上一個程序2的page02;

 

3、將上兩步運行程序1和程序2的瀏覽器關閉但不關閉服務器,重新運行程序2,其結果如下:

步驟三運行結果

對比之前的結果,我們發現page作用域依舊沒有變化,它的值只是程序2即所在頁面裏的值;request作用域僅在當前請求作用,故也以程序2的值爲準,變成page02;session的作用域爲當前會話,因爲前兩步運行程序的瀏覽器關閉了,說明之前的會話都結束了,所以其值恢復成當前的程序2裏的值page02;而application對所有應用有效,也就是只要在應用即服務器還沒重啓清空,都要疊加,即在之前的基礎上再疊加上一個程序2的page02;

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章