轉自:http://blog.csdn.net/jadyer/article/details/6174555html
淺析值棧
ValueStack對象至關於一個棧,它貫穿整個Action的生命週期,每一個Action類的對象實例都會擁有一個ValueStack對象
當Struts2接收到一個*.action請求後,並非直接調用Action方法,而是先將Action類的相應屬性放到ValueStack對象的頂層節點
值棧也位於內存中,它也是和parameters、request、session、application、attr對象放在一塊兒的
值棧屬於ONGL Context裏面的根對象。也就是說它位於整個內存中最最重要的地方,因此叫根對象
根對象和另外五個對象是有區別的,根對象能夠省寫#號,好比<s:property value="user.username"/>
值棧的生命週期與request請求相關,每次請求產生一個值棧。默認全部的Action會被自動放到值棧裏java
服務器跳轉時共用值棧
假設從一個Action11經過服務器跳轉到Action22的話,就意味着這兩個Action是共享一個值棧的,由於一次請求只使用一個值棧
這時內存中狀況是這樣的:首先接收到Action11請求後,會產生一個值棧,在棧頂存放Action11對象以及它全部的屬性
而後通過服務器跳轉到Action22,這時就會把Action22對象壓入值棧的棧頂位置,此時Action11對象以及它的全部屬性就位於棧底了web
取值過程
棧的特徵是後進先出。因而首先到棧頂的對象裏查找是否存在這個屬性,若是棧頂的Action22對象中不存在這個屬性的話
它就會繼續向下尋找直至棧底對象,一直查找是否存在這個屬性
若是最後找到該屬性的話,那麼就會在JSP頁面中經過<s:property value="username"/>輸出屬性值
若是在Action22和Action11都有一個同名的同類型的username屬性的話,那麼將輸出Action22中的屬性值
由於它是先從棧頂開始尋找屬性的,值棧的特徵就是後進先出,但有個前提:請求過程是經過服務器跳轉的apache
三個語法
假設此時想要獲取Action11中的username屬性的話,就可使用值棧的Top語法或者N語法
使用Top語法獲取值棧中的第二個對象的屬性:<s:property value="[1].top.username"/>
使用 N 語法獲取值棧中的第二個對象的屬性:<s:property value="[1].username"/>
另外值棧還有一個@語法,例如使用@語法調用Action中的靜態方法:<s:property value="@vs@getVOMethod()"/>
@vs@get()等價於@vs1@getVOMethod(),指的是棧頂對象的靜態getVOMethod()方法
同理@vs2@getVOMethod()就是取值棧中第二個對象的靜態getVOMethod()方法數組
客戶端跳轉時使用各自的值棧
假如中間某一個步驟中出現了客戶端跳轉的話,那麼兩個Action所使用的就是兩個不一樣的值棧了
因此在Action22中就不能再使用Action11中的屬性了,在最後跳轉到的JSP頁面中也就沒法獲取Action11的屬性了
也即從Action22跳轉到JSP頁面時使用的是redirect的話,那麼最後值棧中是沒有任何的Action對象的
這個時候咱們能夠經過連接傳參,好比<result type="redirect">test.jsp?netname=${username}</result>
意思就是取出Action22中的username屬性做爲參數,經過瀏覽器地址欄傳遞到JSP頁面中
而後使用OGNL中的#號獲取Paraments對象的屬性,即<s:property value="#parameters.netname"/>就能夠取到值了
輔助參考:http://blog.csdn.net/jadyer/archive/2010/09/16/5887509.aspx瀏覽器
手工向值棧中壓入對象
正常狀況下值棧保存的是Action對象,而咱們也能夠直接往值棧中添加其它對象,這時能夠在Action中添加以下代碼
向值棧中添加對象:ActionContext.getContext.getValueStack().push(new Student("沈浪",22));
並且咱們手工往值棧中添加的Student對象會位於棧頂。這是由於Struts2會首先初始化Action,而後才能調用它的方法
初始化Action的時候,便把Action放到值棧中了,而後在執行它的execute()方法時,就又往值棧中添加了Student對象服務器
淺析OGNL
OGNL是Object-Graph Navigation Language的縮寫,是一種功能強大的表達式語言
經過它簡單一致的表達式語法,能夠存取對象的任意屬性,調用對象的方法,遍歷整個對象的結構圖,實現字段類型轉化等功能
OGNL用得最多的地方就是和Struts2的標籤綁定,也能夠在配置文件中經過${}使用OGNL表達式session
OGNL中$號的使用
1..在國際化資源文件中,引用OGNL表達式
2..在struts.xml文件中,引用OGNL表達式app
OGNL中%號的使用
1..使用%{}能夠取出保存在值堆棧中的Action對象,直接調用它的方法
2..若是Action繼承了ActionSupport,那麼在頁面標籤中可使用%{getText('key')}獲取國際化信息
輔助參考:http://blog.csdn.net/jadyer/archive/2010/09/16/5887545.aspx框架
OGNL中#號的使用
OGNL中的#號能夠取出堆棧上下文中存放的對象
名稱 | 做用 | 例子 |
attr | 用於按request>>session>>application順序訪問其屬性 | #attr.userName至關於按順序從三個範圍讀取userName屬性直到找到爲止 |
request | 包含當前HttpServletRequest的屬性的Map | #request.userName至關於request.getAttribute("userName") |
session | 包含當前HttpSession的屬性的Map | #session.userName至關於session.getAttribute("userName") |
application | 包含當前應用的ServletContext的屬性的Map | #application.userName至關於application.getAttribute("userName") |
parameters | 包含當前HTTP請求參數的Map | #parameters.id[0]至關於request.getParameter("id") |
獲取Action中的屬性值或者Action中的對象的某某屬性值
利用<s:property/>標籤能夠直接獲取Action中的引用類型user裏面的username屬性
一樣能夠經過user.address.addr獲取user中引用類型address中的addr屬性的值
像這種一層一層往下傳遞的訪問方式,即所謂的導航,也就是一步步的往下調用
調用Action的對象裏面的普通方法
默認的會把Action放到值棧裏面,而值棧在訪問的時候,並不須要值棧的名字
當咱們調用<s:property value="user.getVOMethod()"/>的時候
它會自動到值棧裏面查找Action對象裏面有沒有user對象,而後它就發現有user
而後它就再找user裏面有沒有getVOMethod()方法,而後它發現有,因而調用getVOMethod()
實際上調用User中的getVOMethod()方法的過程與獲取表單中的姓名密碼的方式都是相同的
都是到值棧裏面查找,找是否存在user對象,若是存在,接着查找user中是否存在某某屬性或方法
調用Action中的靜態方法
一樣咱們也能夠在JSP頁面中寫一個OGNL表達式調用Action中的靜態方法
調用Action中的靜態方法時,與調用user對象的getVOMethod()方法的過程,是大相徑庭的
此時value的寫法是固定的,以@開頭,後面跟上具體的包名,而後@加上靜態方法
好比<s:property value="@com.jadyer.action.LoginAction@getStatic()"/>
另外user對象是LoginAction中的一個屬性,這個屬性會自動的放到值棧裏面
而值棧調用的時候,不用加上@或者包名等等,因此直接user.getVOMethod()就能夠了
調用JDK類中的靜態方法
可使用<s:property value="@@floor(46.58)"/>輸出floor()的執行結果
這就意味着若是不在@@中指定類的話,默認的就表示java.lang.Math類
當前大多數狀況下,咱們都不會省略這個類,都會寫全了的,而後在後面加上靜態方法
集合的僞屬性
OGNL可以引用集合的一些特殊的屬性,這些屬性並非JavaBean模式,例如size()、length()
當表達式引用這些屬性時,OGNL會調用相應的方法,這就是僞屬性
好比獲取List的大小:<s:property value="testList.size"/>
List的僞屬性:size、isEmpty、iterator
Set的僞屬性:size、isEmpty、iterator
Map的僞屬性:size、isEmpty、keys、values
Iterator的僞屬性:next、hasNext
Enumeration僞屬性:next、hasNext、nextElement、hasMoreElements
獲取集合中元素的實質就是調用它的toString()方法
它還能夠直接獲取集合中的元素,事實上是在調用集合的toString()方法
因此咱們能夠根據實際狀況經過重寫集合的toString()方法來實現個性化輸出
甚至它還能夠像訪問數組那樣,直接testList[2]獲取集合中的元素
但這種方法只適用於List,不適用於Map。由於Map的索引是key,不是數值
另外,因爲HashSet中的元素是沒有順序的,因此也不能用下標獲取單個元素
Lambda表達式
補充一下:使用Lambda表達式能夠在OGNL中書寫遞歸式子,在幫助中對它有很詳細的說明
打開幫助中的//struts-2.0.14-all//struts-2.0.14//docs//index.html頁面
在左側的Documentation下面點擊Guides連接,而後在這個頁面中點擊OGNL
最後跳轉到//struts-2.0.14-all//struts-2.0.14//docs//docs//ognl.html
將這個頁面右側的下拉條拖放到最下面,就會看到它的說明了,它舉的例子以下所示
<s:property value="#fib =:[#this==0 ? 0 : #this==1 ? 1 : #fib(#this-2)+#fib(#this-1)], #fib(11)" />
Lambda表達式的語法是:[...] ,中括號前面有一個冒號,全部東西都在中括號裏面寫
也就是說咱們只要看到一個冒號跟着一箇中括號,就表示這裏使用的是Lambda表達式
#this指的是表達式的參數
因此這個例子能夠這樣理解:先判斷這個參數是否等於零,若是等於零,那麼它的值最後就是零
若是參數不等於零,就再判斷它是否等於壹。若是參數等於壹,那麼它的值最後就是壹
若是參數不等於壹,就繼續調用#fib。注意這裏已經用中括號將總體的值賦給了fib
實際上不多可以用獲得Lambda表達式
利用投影獲取屬性
利用投影獲取List中對象的username屬性時,其中{}表示的是一個集合
stus.{username}就表示將suts中全部的username屬性取出組成一個新的列表
利用選擇獲取屬性
OGNL表達式是很靈活的,能夠同時使用選擇技術與投影技術獲取屬性
使用選擇技術時,#this表明當前元素,問號?是把全部知足條件的元素都取出來
上箭頭^是開始的意思,因此stus.{^#this.grade>=60}.{username}輸出的是[張三]
注意,此時輸出文本中包含中括號,這表示它是一個列表
而stus.{?#this.grade>=60}.{username}[0]輸出的是張三,是字符串,兩者是不一樣的
美圓符號$是結束的意思,因此stus.{$#this.grade>=60}.{username}輸出的是[王五]
這三個符合:問號、上箭頭、美圓符所返回的都是List
補充
1..當OGNL取不到值的時候,它不會報錯,而是什麼都不顯示
2..<s:property value="[0]"/>返回的是ValueStack中從上至下的全部的Object
<s:property value="[1]"/>返回的是ValueStack中從上至下的第二個Object
3..<s:property value="[0].username"/>返回的是成員變量username的值
假設ValueStack中存在兩個Action的話,若是第一個Action若是沒有username變量
那麼它會繼續找第二個Action。那麼在什麼狀況下ValueStack中會存在兩個Action呢
答案是在struts.xml中配置的是從一個Action經過<result type="chain">跳轉到另外一個Action
4..<constant name="struts.ognl.allowStaticMethodAccess" value="true"/>
在Struts2.1.6中必須設置struts.ognl.allowStaticMethodAccess爲true以後
才容許使用OGNL訪問靜態方法。而在Struts2.0.11則無需設置,便可直接訪問
下面是OGNL測試的工程代碼,這是一個Struts2.0.11應用
首先是web.xml文件
而後是用於輸入用戶名和密碼等信息的測試頁面login.jsp
而後是用於顯示OGNL處理結果的loginSuc.jsp頁面
而後是struts.xml文件
接着是用到的三個VO類
最後是用來提供OGNL測試的數據的LoginAction.java