struts2之OGNL用法
淺析OGNL
OGNL是Object-GraphNavigation Language的縮寫,是一種功能強大的表達式語言
通過它簡單一致的表達式語法,可以存取對象的任意屬性,調用對象的方法,遍歷整個對象的結構圖,實現字段類型轉化等功能
OGNL用得最多的地方就是和Struts2的標簽綁定,也可以在配置文件中通過${}使用OGNL表達式
?
OGNL中$號的使用
1..在國際化資源文件中,引用OGNL表達式
2..在struts.xml文件中,引用OGNL表達式
?
OGNL中%號的使用
1..使用%{}可以取出保存在值堆棧中的Action對象,直接調用它的方法
2..如果Action繼承了ActionSupport,那么在頁面標簽中可以使用%{getText('key')}獲取國際化信息
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:propertyvalue="user.getVOMethod()"/>的時候
它會自動到值棧里面查找Action對象里面有沒有user對象,然后它就發現有user
然后它就再找user里面有沒有getVOMethod()方法,然后它發現有,于是調用getVOMethod()
實際上調用User中的getVOMethod()方法的過程與獲取表單中的姓名密碼的方式都是相同的
都是到值棧里面查找,找是否存在user對象,如果存在,接著查找user中是否存在某某屬性或方法
?
調用Action中的靜態方法
同樣我們也可以在JSP頁面中寫一個OGNL表達式調用Action中的靜態方法
調用Action中的靜態方法時,與調用user對象的getVOMethod()方法的過程,是截然不同的
此時value的寫法是固定的,以@開頭,后面跟上具體的包名,然后@加上靜態方法
比如<s:propertyvalue="@com.jadyer.action.LoginAction@getStatic()"/>
另外user對象是LoginAction中的一個屬性,這個屬性會自動的放到值棧里面
而值棧調用的時候,不用加上@或者包名等等,所以直接user.getVOMethod()就可以了
?
調用JDK類中的靜態方法
可以使用<s:propertyvalue="@@floor(46.58)"/>輸出floor()的執行結果
這就意味著如果不在@@中指定類的話,默認的就表示java.lang.Math類
當前大多數情況下,我們都不會省略這個類,都會寫全了的,然后在后面加上靜態方法
?
集合的偽屬性
OGNL能夠引用集合的一些特殊的屬性,這些屬性并不是JavaBean模式,例如size()、length()
當表達式引用這些屬性時,OGNL會調用相應的方法,這就是偽屬性
比如獲取List的大小:<s:propertyvalue="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
?
?
淺析值棧
ValueStack對象相當于一個棧,它貫穿整個Action的生命周期,每個Action類的對象實例都會擁有一個ValueStack對象
當Struts2接收到一個*.action請求后,并不是直接調用Action方法,而是先將Action類的相應屬性放到ValueStack對象的頂層節點
值棧也位于內存中,它也是和parameters、request、session、application、attr對象放在一起的
值棧屬于ONGL Context里面的根對象。也就是說它位于整個內存中最最重要的地方,所以叫根對象
根對象和另外五個對象是有區別的,根對象可以省寫#號,比如<s:propertyvalue="user.username"/>
值棧的生命周期與request請求相關,每次請求產生一個值棧。默認所有的Action會被自動放到值棧里
?
服務器跳轉時共用值棧
假設從一個Action11通過服務器跳轉到Action22的話,就意味著這兩個Action是共享一個值棧的,因為一次請求只使用一個值棧
這時內存中情況是這樣的:首先接收到Action11請求后,會產生一個值棧,在棧頂存放Action11對象以及它所有的屬性
然后經過服務器跳轉到Action22,這時就會把Action22對象壓入值棧的棧頂位置,此時Action11對象以及它的所有屬性就位于棧底了
?
取值過程
棧的特征是后進先出。于是首先到棧頂的對象里查找是否存在這個屬性,如果棧頂的Action22對象中不存在這個屬性的話
它就會繼續向下尋找直至棧底對象,一直查找是否存在這個屬性
如果最后找到該屬性的話,那么就會在JSP頁面中通過<s:propertyvalue="username"/>輸出屬性值
如果在Action22和Action11都有一個同名的同類型的username屬性的話,那么將輸出Action22中的屬性值
因為它是先從棧頂開始尋找屬性的,值棧的特征就是后進先出,但有個前提:請求過程是通過服務器跳轉的
?
三個語法
假設此時想要獲取Action11中的username屬性的話,就可以使用值棧的Top語法或者N語法
使用Top語法獲取值棧中的第二個對象的屬性:<s:property value="[1].top.username"/>
使用 N 語法獲取值棧中的第二個對象的屬性:<s:propertyvalue="[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:propertyvalue="#parameters.netname"/>就可以取到值了
?
手工向值棧中壓入對象
正常情況下值棧保存的是Action對象,而我們也可以直接往值棧中添加其它對象,這時可以在Action中添加如下代碼
向值棧中添加對象:ActionContext.getContext.getValueStack().push(new Student("沈浪",22));
而且我們手工往值棧中添加的Student對象會位于棧頂。這是因為Struts2會首先初始化Action,然后才能調用它的方法
初始化Action的時候,便把Action放到值棧中了,然后在執行它的execute()方法時,就又往值棧中添加了Student對象
?
補充
1..當OGNL取不到值的時候,它不會報錯,而是什么都不顯示
2..<s:property value="[0]"/>返回的是ValueStack中從上至下的所有的Object
????<s:propertyvalue="[1]"/>返回的是ValueStack中從上至下的第二個Object
3..<s:property value="[0].username"/>返回的是成員變量username的值
??? 假設ValueStack中存在兩個Action的話,如果第一個Action如果沒有username變量
??? 那么它會繼續找第二個Action。那么在什么情況下ValueStack中會存在兩個Action呢
??? 答案是在struts.xml中配置的是從一個Action通過<resulttype="chain">跳轉到另一個Action
4..<constantname="struts.ognl.allowStaticMethodAccess"value="true"/>
??? 在Struts2.1.6中必須設置struts.ognl.allowStaticMethodAccess為true之后
??? 才允許使用OGNL訪問靜態方法。而在Struts2.0.11則無需設置,即可直接訪問
?
?
eg:User對象屬性獲取 如User中有username和password字段獲取username屬性<s:property value="user.username" />
獲取password屬性<s:property value="user.password" />
若User中又包含定義了address對象,address對象中包含有addr屬性,則可以這樣訪問
獲取addr屬性<s:property value="user.address.addr" />
若User中還包含一個get()的普通方法,可以這樣調用
<s:property value="user.get()" />
以上是調用值棧中對象的普通方法,user為值棧中的對象
調用action中的靜態方法get(),普通方法不能直接調用
<s:property value="@com.netshuai.action.ManagerAction@get()" />
以上為調用非值棧中的靜態方法
調用JDK中類的靜態方法<s:property value="@java.lang.Math@floor(32.56)" />
上例也可寫成<s:property value="@@floor(32.56)" />,省略前面的類則默認使用java.lang.Math類,其他類不可省略
調用普通類中的靜態屬性<s:property value="@com.netshuai.model.Address@city" />
address中的city靜態屬性要用public聲明
調用普通類的構造方法,如構造方法為
public User(String username)
{
??????this.username=username;
}
調用方法為<s:property value="new com.netshuai.model.User('hello').username" />,則返回username值為hello
獲取List<s:property value="list" />
獲取List中的某一個元素<s:property value="list[0]" />
獲取List的大小<s:property value="list.size" />
獲取Set<s:property value="set" />
無法獲取Set中的某一個元素,因為Set沒有順序
獲取Map<s:property value="map" />
獲取Map中所有key的值<s:property value="map.keys" />
獲取Map中所有value的值<s:property value="map.values" />
獲取Map中的某一個元素<s:property value="map['k1']" />
獲取List所有對象<s:property value="listObject" />,需要重寫toString()方法才能正常顯示對象的值?
利用投影獲取List中所有對象的username屬性<s:property value="listObject.{username}" />
利用投影獲取List中第一個對象的username屬性<s:property value="listObject.{username}[0]" />
利用選擇獲取List中年齡大于30的對象<s:property value="listObject.{?#this.age>30}" />
利用選擇獲取List中年齡大于30的對象的username<s:property value="listObject.{?#this.age>30}.{username}" />
利 用選擇獲取List中年齡大于30的第一個對象的username<s:property value="listObject.{?#this.age>30}.{username}[0]" />或<s:property value="listObject.{^#this.age>30}.{username}" />
利用選擇獲取List中年齡大于30的最后一個對象的username<s:property value="listObject.{$#this.age>30}.{username}" />
獲取parameters中的屬性<s:property value="#parameters.name" />
獲取request中的屬性<s:property value="#request.name" />
獲取session中的屬性<s:property value="#session.name" />
獲取application中的屬性<s:property value="#application.name" />
<s:property value="#attr.name" />按順序遍歷上面四個對象,然后返回首先找到的值
%{}可以取出存在值堆棧中的Action對象,直接調用它的方法,如%{getText('key')}可以取出國際化信息
${}可以用在國際化資源文件中和struts2配置文件中
使用top獲取值棧中第二個對象<s:property value="[1].top.user"/>
使用top獲取值棧中第二個對象的屬性<s:property value="[1].user"/>
調用值棧中action的靜態方法get()<s:property value="@vs@get()"/>,vs也可寫做vs1
調用值棧中第二個action的靜態方法get()<s:property value="@vs2@get()"/>
將一個對象放入值棧
ActionContext.getContext().getValueStack().push(user); 總結: 1.當使用OGNL調用靜態方法的時候。需要按照如下語法編寫表達式:@package.classname@methodname(parameter)
2.對于OGNL來說。java.lang.Math是其的默認類。如果調用java.lang.Math的靜態方法時。無需指定類的名字。比如@@min(4,10)
3.對于OGNL來說,數組和集合是一樣的。都是通過下標索引來訪問的。構造集合的時候用{...}形式
4.使用OGNL來映射(Map)的語法格式如下所示:#{'key1':'value1','key2','value2'}
5.過濾(filtering):conllection.{?,表達式}這是針對集合來處理的。當“?”被"^"代替時表示獲取集合中的第一個對象。當“?”被"$"代替時表示獲取集合中的最后一個對象。
6.OGNL針對集合提供了一些偽屬性(如size,isEmpty),讓我們可以通過屬性的方式來調用方法。(本質原因在于集合當中的很多方法并不符合javaBean的命名規則),但我們依然可以通過調用方法來實現與偽屬性相同的目的
7.在使用過濾操作的時候。我們通常使用#this.該表達式用于代表當前正在迭代的集合中的對象
8.過濾(projection):collection.{expression}
9.過濾與投影的區別:類比于數據庫中的表。過濾是取行的操作,而投影是取列的操作。
10.在struts2中有個稱之為值棧的概念(valueStack)
11.在struts2中,根對象就是valueStack,在struts2的任何流程中。valueStack中的最頂層對象一定是Action對象。
12.訪問靜態方法或是靜態成員變量的:@vs@method
13.關于struts2標簽庫屬性值的%和#號的關系,如果標簽的屬性值是OGNL表達式,那么無需加上%{} 如果 標簽的屬性值的是字符串類型的。那么在字符串當中凡是出現%{}都會被解析成OGNL表達式,解析完畢后再與其他的字符串進行拼接構造出最后的字符串值。我們可以再所有的屬性值加上%{},這樣如果該屬性值是OGNL表達式,那么標簽處理類就會將%{}忽略掉
轉載于:https://www.cnblogs.com/sxxjyj/p/6093405.html
總結
以上是生活随笔為你收集整理的struts2之OGNL用法的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 在ueditor编辑器的光标停留处插入内
- 下一篇: 微积分5--隐函数