Struts2的值棧和物件棧
筆者不知道該用哪個詞來形容ValueStack、ActionContext等可以在Struts2中用來存放資料的類。這些類使用的範圍不同,得到的方法也不同,下面就來一一介紹。
宣告:本文參考Struts2版本為2.3.1.2,內容僅供參考,限於筆者水平有限,難免有所疏漏,望您能友善指出。本文發表於ITEYE,謝絕轉載。
1. ValueStack
ValueStack在中文版的《Struts2深入淺出》一書中譯作“值棧”。其本身資料結構是一個棧,使用者可以把一些物件(又稱作bean)存入值棧中,然後使用動態的表示式來讀取bean的屬性,或者對bean進行一些其他操作。由於值棧中可能有多個bean,值棧會按bean出棧的順序依次嘗試使用動態的表示式來讀取值,直到成功讀取值為止。在Struts2中,預設的值棧實現是
在Struts2執行一次請求的過程中,Struts2會把當前的Action物件自動放入值棧。這樣,在渲染JSP時,JSP裡的程式碼使用<s:property value="..."/>之類標籤中的Ognl表示式會直接作用於Action物件,從而方便的讀取Action的屬性。
如何得到值棧:在自定義的攔截器中,使用ActionInvocation.getStack()方法( ActionInvocation 是攔截器的方法引數)。
在Action類中,讓攔截器注入ValueStack或者使用ActionContext
在JSP中,直接使用標籤即可獲得值棧裡的資料,而一般不用獲取值棧本身。
Struts2自動存入Action:之前已經提到,Struts2在執行一次請求的過程中會把當前的Action物件自動存入值棧中。
ModelDrivenInterceptor會存入Action的model屬性:如果你使用了Struts2提供的 ModelDrivenInterceptor,則它會把Action物件的getModel()方法得到的物件存入值棧中。此時,值棧最底層為Action類,其次為這個model。
在自定義的攔截器中存入值棧:得到值棧物件後呼叫ValueStack.put(Object object)方法。
在Action類中存入值棧:得到值棧物件後呼叫ValueStack.put(Object object)方法。
在JSP中存入值棧:標籤<s:push value="..."></s:push>是專門用來在JSP中把指定的value放入值棧的,但value被放入值棧的時間僅在s:push標籤內,即程式執行到</s:push>標籤處會把value從值棧中移出。另外,還有一些標籤比如<s:iterator/>由於其功能的需要也會把一些物件放到值棧中。
在自定義的攔截器中,獲得值棧後,使用ValueStack.findValue(...)等方法。
在Action類中,獲得值棧後,使用ValueStack.findVlaue(...)等方法。
在JSP中,一些標籤的屬性是直接在值棧上執行Ognl表示式的,比如<s:property/>的value屬性。如果標籤的屬性不是直接執行Ognl表示式的,則需要使用“%{}”將表示式括起來,這樣Struts2就會以Ognl表示式來執行了。至於到底哪些標籤是直接執行Ognl而哪些不是,請參考完整的官方文件。
在JSP中,使用[0]、[1]等表示式來指定從棧的第幾層開始執行表示式。[0]表示從棧頂開始,[1]表示從棧的第二層開始。比如表示式“name”等價於“[0].name”。參見此處。
在表示式中使用top關鍵字來訪問物件本身。比如,表示式“name”等價於“top.name”,表示式“[0].top”等價於“top”,表示式“[1].top.name”等價於“[1].name”。
總之,值棧主要目的是為了讓JSP內能方便的訪問Action的屬性。
一些例子:
1 // 此類為一個封裝資料的簡單類,在下面的例子會用到
2 public class Person {
3
4 private String name;
5
6 public String getName() {
7 return name;
8 }
9
10 public void setName(String name) {
11 this.name = name;
12 }
13 }
1 // 本類將演示攔截器中對值棧的操作
2 public class MyInterceptor extends AbstractInterceptor {
3
4 public String intercept(ActionInvocation invocation) throws Exception {
5 // 獲得值棧
6 ValueStack valueStack = invocation.getStack();
7 // 存入值
8 Person person = new Person();
9 valueStack.push(person);
10 // 執行表示式獲取值
11 String name = (String) valueStack.findValue("name");
12 // 其他程式碼
13 return invocation.invoke();
14 }
15 }
1 // 本類將演示在Action中對值棧進行操作
2 public class MyAction extends ActionSupport {
3
4 @Override
5 public String execute() throws Exception {
6 // 獲得值棧
7 ValueStack valueStack = ActionContext.getContext().getValueStack();
8 // 存入值
9 Person person = new Person();// 這是之前例子中定義的類
10 valueStack.push(person);
11 // 執行表示式獲取值
12 String name = (String) valueStack.findValue("name");
13 // 其他程式碼
14 // ......
15 return SUCCESS;
16 }
17 // 以下定義的屬性供接下來的JSP例子使用
18 private String message;
19 private Person person;
20 private List<Person> personList;
21
22 public String getMessage() {
23 return message;
24 }
25
26 public Person getPerson() {
27 return person;
28 }
29
30 public List<Person> getPersonList() {
31 return personList;
32 }
33 }
1 <%@page contentType="text/html" pageEncoding="UTF-8"%>
2 <%@taglib uri="/struts-tags" prefix="s" %>
3 <!DOCTYPE html>
4 <html>
5 <head>
6 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
7 <title>JSP Page</title>
8 </head>
9 <body>
10 <!-- 本JSP將演示在JSP中對值棧的使用 -->
11 <!-- 本JSP為MyAction對應的JSP -->
12
13 <!-- 由於Action已經被存入的值棧,所以可以呼叫Action的屬性 -->
14 <!-- 使用下面的標籤和表示式來顯示MyAction的message屬性 -->
15 <s:property value="message"/>
16 <!-- 使用下面的標籤和表示式來呼叫Action的getText(...)方法,引數為MyAction的message屬性 -->
17 <s:property value="getText(message)"/>
18 <!-- 預設情況下傳遞給cssClass的是字串常量。可以使用“%{}”來啟用Ognl,這樣,傳遞給cssClass的就不是字串常量"message",而是上面所說的message的值 -->
19 <s:div cssClass="%{message}"/>
20 <!-- 使用s:push標籤來將物件放入值棧,如下 -->
21 <s:push value="person">
22 <!-- 在此s:push標籤內,值棧的棧頂元素為person,棧頂第二層為action
23 <!-- 在標籤內直接呼叫person的屬性(而不是Action的屬性),如下 -->
24 <s:property value="name"/>
25 <!-- 在標籤內也可以使用MyAction的屬性,值棧會依次先查詢Person是否有該屬性,由於沒找到,會再MyAction中再查詢,如下 -->
26 <s:property value="message"/>
27 <!-- 可以使用“[0]”、“[1]”等指定從值棧的哪一層開始查詢 -->
28 <!-- 此時,使用“[0]”表示從Person開始查詢,當然還是找不到,值棧就接著到MyAction中查詢,如下 -->
29 <s:property value="[0].message"/>
30 <!-- 此時,使用“[1]”將從MyAction開始查詢,而跳過了person,如下 -->
31 <s:property value="[1].message"/>
32 <!-- 想要訪問棧頂元素本身使用關鍵字“top”,比如,下面的top就代表棧頂的person,如下 -->
33 <s:property value="top"/>
34 <!-- 或者如下 -->
35 <s:property value="[0].top"/>
36 <!-- 想要訪問MyAction本身的話使用如下寫法 -->
37 <s:property value="[1].top"/>
38 </s:push>
39 <!-- 此時person已被移出值棧,再使用如下標籤和表示式將無法得到值 -->
40 <!--<s:property value="name"/>-->
41 <!-- iterator標籤會把list的每個元素依次存入棧頂,如下 -->
42 <s:iterator value="personList">
43 <!-- 獲得List每個元素中的name屬性 -->
44 <s:property value="name"/>
45 </s:iterator>
46 </body>
47 </html>
2.ActionContext
ActionContext是Action的上下文,Struts2自動在其中儲存了一些在Action執行過程中所需的物件,比如session, parameters, locale等。Struts2會根據每個執行HTTP請求的執行緒來建立對應的ActionContext,即一個執行緒有一個唯一的ActionContext。因此,使用者可以使用靜態方法ActionContext.getContext()來獲取當前執行緒的ActionContext,也正是由於這個原因,使用者不用去操心讓Action是執行緒安全的。
無論如何,ActionContext都是用來存放資料的。Struts2本身會在其中放入不少資料,而使用者也可以放入自己想要的資料。ActionContext本身的資料結構是對映結構,即一個Map,用key來對映value。所以使用者完全可以像使用Map一樣來使用它,或者直接使用Action.getContextMap()方法來對Map進行操作。
Struts2本身在其中放入的資料有ActionInvocation、application(即ServletContext)、conversionErrors、Locale、action的name、request的引數、HTTP的Session以及值棧等。完整的列表請參考它的Javadoc(本文附錄有對它包含內容的討論)。
由於ActionContext的執行緒唯一和靜態方法就能獲得的特性,使得在非Action類中可以直接獲得它,而不需要等待Action傳入或注入。需要注意的是,它僅在由於request而建立的執行緒中有效(因為request時才建立對應的ActionContext),而在伺服器啟動的執行緒中(比如fliter的init方法)無效。由於在非Action類中訪問其的方便性,ActionContext也可以用來在非Action類中向JSP傳遞資料(因為JSP也能很方便的訪問它)。
ValueStack與ActionContext的聯絡和區別:相同點:它們都是在一次HTTP請求的範圍內使用的,即它們的生命週期都是一次請求。
不同點:值棧是棧的結構,ActionContext是對映(Map)的結構。
聯絡:ValueStack.getContext()方法得到的Map其實就是ActionContext的Map。檢視Struts2的原始碼可知(Struts2.3.1.2的org.apache.struts2.dispatcher.ng.PrepareOperations的第79行,createActionContext方法),在建立ActionContext時,就是把ValueStack.getContext()作為ActionContext的建構函式的引數。所以,ValueStack和ActionContext本質上可以互相獲得。
注意:在一些文件中,會出現把物件存入“stack's context”的字樣,其實就是把值存入了ActionContext。所以在閱讀這些文件時,要看清楚,到底是放入了棧結構(即值棧),還是對映結構(值棧的context,即ActionContext)。
在自定義的攔截器中:使用ActionInvocation.getInvocationContext()或者使用ActionContext.getContext()。
在Action類中:讓攔截器注入或者使用ActionContext.getContext()。
在非Action類中:讓Action類傳遞引數、使用注入機制注入或者使用ActionContext.getContext()。注意:只有執行在request執行緒中的程式碼才能呼叫ActionContext.getContext(),否則返回的是null。
在JSP中:一般不需要獲得ActionContext本身。
在攔截器、Action類、非Action類等Java類中:使用ActionContext.put(Object key, Object value)方法。
在JSP中:標籤<s:set value="..."/>預設將值存入ActionContext中(當然,<s:set>標籤還可以把值存到其他地方)。另外,許多標籤都有var屬性(以前用的是id屬性,現在id屬性已被棄用),這個屬效能向ActionContext存入值,key為var屬性的值,value為標籤的value屬性的值。(有些文件寫的是向ValueStack的context存入值,其實是一樣的)
在攔截器、Action類、非Action類等Java類中:使用ActionContext.get(Object key)方法。
在JSP中:使用#開頭的Ognl表示式,比如<s:property value="#name"/>會呼叫ActionContext.get("name")方法。注意:如果某標籤的屬性預設不作為Ognl表示式解析,則需要使用%{}把表示式括起來,於是就會出現類似“%{#name}的表示式”。(“#”的更多用途參見這裡)
總之,在JSP中使用ActionContext一方面是由於它是對映結構,另一方面是能讀取Action的一些配置。當你需要為許多Action提供通用的值的話,可以讓每個Action都提供getXXX()方法,但更好的方法是在攔截器或JSP模板中把這些通用的值存放到ActionContext中(因為攔截器或JSP模板往往通用於多個Action)。
一些例子:
1 // 本類將演示攔截器中對ActionContext的操作
2 public class MyInterceptor extends AbstractInterceptor {
3
4 public String intercept(ActionInvocation invocation) throws Exception {
5 // 獲得ActionContext
6 ActionContext actionContext = invocation.getInvocationContext();
7 // 存入值
8 Person person = new Person();
9 actionContext.put("person", person);
10 // 獲取值
11 Object value = actionContext.get("person");
12 // 獲取HttpServletRequest
13 HttpServletRequest request = (HttpServletRequest) actionContext.get(StrutsStatics.HTTP_REQUEST);
14 // 獲取request的Map,即HttpServletRequest.getAttribute(...)和HttpServletRequest.setAttribute(...)所操作的值
15 Map requestMap = (Map) actionContext.get("request");
16 // 其他程式碼
17 // ......
18 return invocation.invoke();
19 }
20 }
1 // 本類將演示在Action中對ActionContext進行操作
2 public class MyAction extends ActionSupport {
3
4 @Override
5 public String execute() throws Exception {
6 // 獲得值棧
7 ActionContext actionContext = ActionContext.getContext();
8 // 存入值
9 Person person = new Person();// 這是之前例子中定義的類
10 actionContext.put("person", person);
11 // 獲取值
12 Object object = actionContext.get("person");
13 // 其他程式碼
14 // ......
15 return SUCCESS;
16 }
17 }
1 <!DOCTYPE html>
2 <html>
3 <head>
4 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
5 <title>JSP Page</title>
6 </head>
7 <body>
8 <!-- 本JSP將演示在JSP中對ActionContext的使用 -->
9 <!-- 本JSP為MyAction對應的JSP -->
10
11 <!-- 由於Action中已經向ActionContext存入了key為"person"的值,所以可以使用“#person”來獲取它,如下 -->
12 <s:property value="#person"/>
13 <!-- 獲得person的name屬性,如下 -->
14 <s:property value="#person.name"/>
15 <!-- 獲得Struts2在ActionContext中存入的值,比如request的Map,如下 -->
16 <s:property value="#request"/>
17 <!-- 獲得Struts2在ActionContext中存入的值,比如session的Map,如下 -->
18 <s:property value="#session"/>
19 <!-- 獲得Struts2在ActionContext中存入的值,request請求傳遞的GET引數或POST引數的Map,如下 -->
20 <s:property value="#parameters"/>
21
22 <!-- 以下演示在JSP中把值存入ActionContext中 -->
23 <!-- 存入一個字串"myName",key為"myKey",如下 -->
24 <s:set value="%{'myName'}" var="myKey"/>
25 <!-- 使用s:bean標籤來建立一個物件,並把它存入ActionContext中,key為myObject,如下 -->
26 <s:bean name="com.example.Person" var="myObject"/>
27 <!-- 之後就可以用“#”來讀取它們,如下 -->
28 <s:property value="#myKey"/>
29 <s:property value="#myObject"/>
30 </body>
31 </html>
3. HttpServletRequest類或request的Map
Struts2中提供了兩種對request的操作:一種是Web伺服器提供的HttpServletRequest類,這和傳統Java Web專案中的操作request的方式相同;另一種是一個“request的Map”,即封裝了HttpServletRequest的attributes的對映類,操作該Map相當於操作HttpServletRequest的attributes。之所以提供了Map的操作方式,一是方便操作,二是能方便使用Ognl在JSP標籤中讀取request。無論如何,這兩個request是互通的。至於request的生命週期等概念,與其他的Java Web專案沒有區別,本文不再詳述。
使用HttpServletRequest類還是request的Map雖然兩者是互通的,但就讀取request的attributes而言,使用request的Map要方便許多,並且不會暴露不必要的介面。當然,HttpServletRequest有一些request的Map沒有的方法,使用這些方法時當然還是要用前者。
兩者都是Map,兩者的生命週期都是一個請求。
傳統的Java Web專案中,往往是通過request的attributes來向JSP傳遞值的:先在Servlet裡setAttribute(),然後在JSP裡getAttribute()。當然在Struts2的專案中,你仍然可以使用這個方法,然而拋棄了Struts2提供的傳遞功能是得不償失的。雖然筆者沒有找到官方文件說一定要用ActionContext替換request的Map,也沒有發現程式中有能獲得ActionContext卻獲得不了request的Map的地方,但在Struts2框架下,操作ActionContext要比操作request的Map更加方便。因此,筆者建議:儘量使用ActionContext而不是request的Map來傳遞值。
request的Map有時候會包含其他框架設定的值,比如spring框架。獲取這些值的時候就需要用request的Map了,因為ActionContext裡沒有。
通過ActionContext可以獲得HttpServletRequest類:“HttpServletRequest request = (HttpServletRequest) actionContext.get(StrutsStatics.HTTP_REQUEST);”。
通過ActionContext也可以獲得request的Map:“Map requestMap = (Map) actionContext.get("request");”。因此,在JSP標籤中,使用表示式“#request”就可以獲得request的Map的資料。
如果已經有ActionContext,則使用“actionContext.get(StrutsStatics.HTTP_REQUEST)”來獲得HttpServletRequest。
在自定義的攔截器中,先獲得ActionContext,再通過ActionContext來獲得。
在Action中,先獲得ActionContext,再通過ActionContext來獲得。或者讓Action實現ServletRequestAware介面,並使用ServletConfigInterceptor攔截器,這樣這個攔截器就會注入HttpServletRequest。
在JSP中,一般不需要獲得HttpServletRequest。
如果已經有ActionContext,則使用“actionContext.get("request")”來獲得。
在自定義的攔截器中,先獲得 ActionContext,再通過ActionContext來獲得。
在Action中,先獲得ActionContext,再通過ActionContext來獲得。或者讓Action實現RequestAware介面,並使用ServletConfigInterceptor攔截器,這樣這個攔截器就會注入Map request。
在JSP中,用“#request”來獲得request的Map,用“#request.key”或者“#request['key']”來讀取Map中的值。
總之,request仍然符合Java Web網站的一般規律。不過筆者建議使用者應儘量避免用request傳值。
一些例子:
1 // 本類將演示攔截器中對HttpServletRequest和request的Map的操作
2 public class MyInterceptor extends AbstractInterceptor {
3
4 public String intercept(ActionInvocation invocation) throws Exception {
5 // 獲得ActionContext
6 ActionContext actionContext = invocation.getInvocationContext();
7 // 獲得HttpServletRequest
8 HttpServletRequest httpServletRequest=(HttpServletRequest)actionContext.get(StrutsStatics.HTTP_REQUEST);
9 // 獲得request的Map
10 Map requestMap = (Map) actionContext.get("request");
11 // 建立一個類作為例項
12 Person person = new Person();
13 // 以下兩行的語句作用相同
14 httpServletRequest.setAttribute("person", person);
15 requestMap.put("person", person);
16 // 其他程式碼
17 // ......
18 return invocation.invoke();
19 }
20 }
1 // 本類將演示在Action中對HttpServletRequest和request的Map進行操作(靜態方法獲得ActionContext)
2 public class MyAction extends ActionSupport {
3
4 @Override
5 public String execute() throws Exception {
6 // 獲得ActionContext
7 ActionContext actionContext = ActionContext.getContext();
8 // 獲得HttpServletRequest
9 HttpServletRequest httpServletRequest=(HttpServletRequest)actionContext.get(StrutsStatics.HTTP_REQUEST);
10 // 獲得request的Map
11 Map requestMap = (Map) actionContext.get("request");
12 // 建立一個類作為例項
13 Person person = new Person();
14 // 以下兩行的語句作用相同
15 httpServletRequest.setAttribute("person", person);
16 requestMap.put("person", person);
17 // 其他程式碼
18 // ......
19 return SUCCESS;
20 }
21 }
1 // 本類將演示在Action中使用ServletRequestAware獲得HttpServletRequest(注意:要使用ServletConfigInterceptor攔截器)
2 public class MyAction extends ActionSupport implements ServletRequestAware {
3
4 private HttpServletRequest request;
5
6 //此方法是介面ServletRequestAware的方法
7 public void setServletRequest(HttpServletRequest request) {
8 this.request = request;
9 }
10
11 @Override
12 public String execute() throws Exception {
13 // HttpServletRequest已在該類的欄位中準備好,可直接使用
14 // ......
15 return SUCCESS;
16 }
17 }
1 <!DOCTYPE html>
2 <html>
3 <head>
4 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
5 <title>JSP Page</title>
6 </head>
7 <body>
8 <!-- 本JSP將演示在JSP中對request的Map的使用 -->
9 <!-- 本JSP為MyAction對應的JSP -->
10
11 <!-- request的Map是Struts2自動在ActionContext中存入的值(key為request),所以使用“#”來訪問ActionContext,從中讀取request -->
12 <s:property value="#request"/>
13 <!-- 以下兩行均是訪問request的Map中key為“name”的值 -->
14 <s:property value="#request.name"/>
15 <s:property value="#request['name']"/>
16 </body>
17 </html>
1 // 本類將演示在Action中使用ServletRequestAware獲得request的Map(注意:要使用ServletConfigInterceptor攔截器)
2 public class MyAction extends ActionSupport implements RequestAware {
3
4 Map<String, Object> request;
5
6 // 該方法是介面RequestAware的方法
7 public void setRequest(Map<String, Object> request) {
8 this.request = request;
9 }
10
11 @Override
12 public String execute() throws Exception {
13 // request的Map已在該類的欄位中準備好,可直接使用
14 // ......
15 return SUCCESS;
16 }
17 }
3. Parameters,即GET請求或POST請求的引數
Parameters為GET或POST等請求時瀏覽器向伺服器傳遞而來的引數。在傳統的Java Web專案中,使用HttpServletRequest.getParameter()等方法來獲取引數,並且可以直接使用HttpServletRequest.getParameterMap()來獲得一個封裝了引數的Map。而在Struts2中,Struts2直接把上述Map存放到了ActionContext中,key為“parameters”。另外,ActionContext還直接提供了ActionContext.getParameters()方法來獲得這個Map。因此,在Struts2的各個部件中操作parameters的方法和操作request的Map的方法十分相似,本段不再詳述。
4. HttpServletSession類和session的Map
傳統Java Web專案中的session是我們都熟悉的,我們用它來記錄一個使用者的會話狀態。Struts2把HttpServletSession封裝到了一個Map中,即“session的Map”,這類似對request的處理。然而為了節省系統資源,我們在不需要session的時候不會建立session。可能正是因為這個緣故,Struts2中沒有把HttpServletSession放入ActionContext中,如果你的程式需要使用HttpServletSession,應該先獲得HttpServletRequest,然後使用getSession()或getSession(boolean b)來獲得它,同時決定是否需要建立session。對於session的Map,Struts2仍然把它放入了ActionContext中(key為"session"),但是不要擔心,這個Map的機制使得只有put新值時才會建立session。總之,Struts2中對HttpServletSession的操作要先通過HttpServletRequest來獲得它,而對session的Map的操作與對request的Map的操作如出一轍,本段不再詳述。
5. ServletContext和application的Map
傳統的Java Web專案中,ServletContext用來存放全域性變數,每個Java虛擬機器每個Web專案只有一個ServletContext。這個ServletContext是由Web伺服器建立的,來保證它的唯一性。ServletContext有一些方法能操作它的attributes,這些操作方法和操作一個Map類似。於是,Struts2又來封裝了:它把ServletContext的attributes封裝到了一個Map中,即“application的Map”,並且也放入的ActionContext中(key為application),因此,對application的Map的操作就如果對request的Map操作,本段不再詳述。
至於對ServletContext的操作,與HttpServletRequest的操作類似:Struts2將ServletContext放到了 ActionContext中,並且ServletConfigInterceptor提供了對ServletContext的注入介面ServletContextAware。因此,本段不再詳述。
注意:在Ognl表示式中使用“#application”可以得到application的Map,而不是ServletContext。然而在JSP嵌入的Java程式碼中(比如“<% application.getAttribute(""); %>”),application為ServletContext,而不是Map。
用一張表格來總結:
變數 | 從ActionContext中獲得 | 生命週期 | 用Ongl來讀取值 | |
ActionContext類 | 靜態方法ActionContext. getContext() | 一次Http請求 | 使用“#”加上key,如“#name” | 無法注入 |
ValueStack類 | ActionContext. getValueStack() | 一次Http請求 | 直接填寫來訪問棧中物件的方法,或者使用top來直接獲得棧中物件 | 無法注入 |
HttpServletRequest類 | ActionContext. get( StrutsStatics. HTTP_REQUEST) | 一次Http請求 | 無方便的方法 | |
request的Map | ActionContext. get("request") | 一次Http請求 | 使用“#request”再加上key,如“#request.name”或者“#request['name']” | |
parameters的Map | ActionContext. get( "parameters") | 一次Http請求 | 使用“# parameters”再加上key,如“#parameters .name”或者“#parameters ['name']” | |
HttpServletSession類 | 無(需通過HttpServletRequest來獲得) | 一次Http Session會話 | 無方便的方法 | 無法注入 |
session的Map | ActionContext. get("session") | 每次請求建立,但在一次Http Session會話中資料都是一樣的 | 使用“#session”再加上key,如“# session.name”或者“#session ['name']” | |
ServletContext類 | ActionContext. get( StrutsStatics. SERVLET_CONTEXT) | 網站專案啟動後一直存在且唯一 | 無方便的方法 | |
application的Map | ActionContext.get( "application") | 每次請求時建立,但其中的資料是網站專案啟動後一直存在且共享 | 使用“# application”再加上key,如“#application .name”或者“#application ['name']” |
key |