ognl表達式

OGNL表達式語言

OGNL是Object Graphic Navigation Language(對象圖導航語言)的縮寫,它是一個開源項目。 Struts 2框架使用OGNL作爲默認的表達式語言。

Ognl 有一個上下文(Context)概念,說白了上下文就是一個MAP結構,它實現了java.utils.Map接口,在Struts2中上下文(Context)的實現爲ActionContext,下面是上下文(Context)的結構示意圖

 

Struts 2中的OGNL Context實現者爲ActionContext,它結構示意圖如下:

                               |--request  

                               |--application  

OGNL  context ------ | ValueStack(值棧,它是根對象)   

                               |--session  

                               |--attr  

                               |--parameters

當Struts2接受一個請求時,會迅速創建ActionContext,ValueStack,action 。然後把action存放進ValueStack,所以action的實例變量可以被OGNL訪問。

 

訪問上下文(Context)中的對象需要使用#符號標註命名空間,如#application、#session

另外OGNL會設定一個根對象(root對象),在Struts2中根對象就是ValueStack(值棧) 。如果要訪問根對象(即ValueStack)中對象的屬性,則可以省略#命名空間,直接訪問該對象的屬性即可

在struts2中,根對象ValueStack的實現類爲OgnlValueStack,該對象不是我們想像的只存放單個值,而是存放一組對象。在OgnlValueStack類裏有一個List類型的root變量,就是使用他存放一組對象

                   |--request  

                   |--application  

context ------|--OgnlValueStack root變量[action, OgnlUtil, ... ]  

                   |--session  

                   |--attr  

                   |--parameters

在root變量中處於第一位的對象叫棧頂對象。通常我們OGNL表達式裏直接寫上屬性的名稱即可訪問root變量裏對象的屬性,搜索順序是從棧頂對象開始尋找,如果棧頂對象不存在該屬性,就會從第二個對象尋找,如果沒有找到就從第三個對象尋找,依次往下訪問,直到找到爲止。

大家注意: Struts2中,OGNL表達式需要配合Struts標籤纔可以使用。如:<s:property value="name"/>

由於ValueStack(值棧)是Struts 2中OGNL的根對象,如果用戶需要訪問值棧中的對象,在JSP頁面可以直接通過下面的EL表達式訪問ValueStack(值棧)中對象的屬性:

${foo} //獲得值棧中某個對象的foo屬性

如果訪問其他Context中的對象,由於他們不是根對象,所以在訪問時,需要添加#前綴。

l  application對象:用於訪問ServletContext,例如#application.userName或者#application['userName'],相當於調用ServletContext的getAttribute("username")。

l  session對象:用來訪問HttpSession,例如#session.userName或者#session['userName'],相當於調用session.getAttribute("userName")。

l  request對象:用來訪問HttpServletRequest屬性(attribute)的Map,例如#request.userName或者#request['userName'],相當於調用request.getAttribute("userName")。

l  parameters對象:用於訪問HTTP的請求參數,例如#parameters.userName或者#parameters['userName'],相當於調用request.getParameter("username")。

l  attr對象:用於按page->request->session->application順序訪問其屬性。

爲何使用EL表達式能夠訪問valueStack中對象的屬性

原因是Struts2對HttpServletRequest作了進一步的封裝。簡略代碼如下:

 public class StrutsRequestWrapper extends HttpServletRequestWrapper {

       public StrutsRequestWrapper(HttpServletRequest req) {

           super(req);

       }

       public Object getAttribute(String s) {

        ......

        ActionContext ctx = ActionContext.getContext();

        Object attribute = super.getAttribute(s);//先從request範圍獲取屬性值

        if (ctx != null) {

            if (attribute == null) {//如果從request範圍沒有找到屬性值,即從ValueStack中查找對象的屬性值

               ......

               ValueStack stack = ctx.getValueStack();

             attribute = stack.findValue(s);

               ......

            }

        }

        return attribute;

    }

 }

採用OGNL表達式創建List/Map集合對象

如果需要一個集合元素的時候(例如List對象或者Map對象),可以使用OGNL中同集合相關的表達式。

使用如下代碼直接生成一個List對象:

 <s:set name="list" value="{'zhangming','xiaoi','liming'}" />

<s:iterator value="#list" id="n">

      <s:property value="n"/><br>

</s:iterator>

生成一個Map對象:

<s:set name="foobar" value="#{'foo1':'bar1''foo2':'bar2'}" />

<s:iterator value="#foobar" >

      <s:property value="key"/>=<s:property value="value"/><br>

</s:iterator>

 

Set標籤用於將某個值放入指定範圍。

scope:指定變量被放置的範圍,該屬性可以接受application、session、request、 page或action。如果沒有設置該屬性,則默認放置在OGNL Context中(#訪問)

value:賦給變量的值.如果沒有設置該屬性,則將ValueStack棧頂的值賦給變量。

對於集合類型,OGNL表達式可以使用in和not in兩個元素符號。其中,in表達式用來判斷某個元素是否在指定的集合對象中;not in判斷某個元素是否不在指定的集合對象中,如下所示。

採用OGNL表達式判斷對象是否存在於集合中

in表達式:

<s:if test="'foo' in {'foo','bar'}">

   在

</s:if>

<s:else>

   不在

</s:else>

not in表達式:

<s:if test="'foo' not in {'foo','bar'}">

   不在

</s:if>

<s:else>

   在

</s:else>

 

OGNL表達式的投影功能

除了in和not in之外,OGNL還允許使用某個規則獲得集合對象的子集,常用的有以下3個相關操作符。

?:獲得所有符合邏輯的元素。

^:獲得符合邏輯的第一個元素。

$:獲得符合邏輯的最後一個元素。

例如代碼:

<s:iterator value="books.{?#this.price > 35}">

      <s:property value="title" /> - $<s:property value="price" /><br>

</s:iterator>

在上面代碼中,直接在集合後緊跟.{}運算符表明用於取出該集合的子集{}內的表達式用於獲取符合條件的元素this指的是爲了從大集合books篩選數據到小集合,需要對大集合books進行迭代,this代表當前迭代的元素。本例的表達式用於獲取集合中價格大於35的書集合。

public class BookAction extends ActionSupport {

      private List<Book> books;

      ....

      @Override

          public String execute() {

                   books = new LinkedList<Book>();

                   books.add(new Book("A735619678", "spring", 67));

           books.add(new Book("B435555322", "ejb3.0",15));

      }

}

 

property標籤

property標籤用於輸出指定值:

<s:set name="name" value="'kk'" />

<s:property value="#name"/>

default:可選屬性,如果需要輸出的屬性值爲null,則顯示該屬性指定的值

escape:可選屬性,指定是否格式化HTML代碼

value:可選屬性,指定需要輸出的屬性值,如果沒有指定該屬性,則默認輸出ValueStack棧頂的值

 

iterator標

iterator標籤用於對集合進行迭代,這裏的集合包含List、Set和數組。會把當前迭代的標籤置於棧頂

<s:set name="list" value="{'zhangming','xiaoi','liming'}" />

<s:iterator value="#liststatus="st"><!--st放在上下文中,用#訪問-->

      <font color=<s:if test="#st.odd">red</s:if><s:else>blue</s:else>>

      <s:property /></font><br>

</s:iterator>

value:可選屬性,指定被迭代的集合,如果沒有設置該屬性,則使用ValueStack棧頂的集合

id:可選屬性,指定集合裏元素的id。

status:可選屬性,該屬性指定迭代時的IteratorStatus實例。該實例包含如下幾個方法:

      int getCount(),返回當前迭代了幾個元素。

      int getIndex(),返回當前迭代元素的索引。

      boolean isEven(),返回當前被迭代元素的索引是否是偶數

      boolean isOdd(),返回當前被迭代元素的索引是否是奇數

      boolean isFirst(),返回當前被迭代元素是否是第一個元素。

      boolean isLast(),返回當前被迭代元素是否是最後一個元素。

if/elseif/else標籤

<s:set name="age" value="21" />

<s:if test="#age==23">

      23

</s:if>

<s:elseif test="#age==21">

      21

</s:elseif>

<s:else>

      都不等

</s:else>

 

url標籤

<s:url action="helloworld_add" namespace="/test"><s:param name="personid" value="23"/></s:url>

生成類似如下路徑:

/struts/test/helloworld_add.action?personid=23

 

當標籤的屬性值作爲字符串類型處理時, %”符號的用途是計算OGNL表達式的值

  <s:set name="myurl" value="'http://www.foshanshop.net'"/>

   <s:url value="#myurl" /><br>

   <s:url value="%{#myurl}" />

輸出結果:

#myurl

http://www.foshanshop.net

 

表單標籤_checkboxlist複選框

如果集合爲list

<s:checkboxlist name="list" list="{'Java','.Net','RoR','PHP'}" value="{'Java','.Net'}"/>

生成如下html代碼:

<input type="checkbox" name="list" value="Java" checked="checked"/><label>Java</label>

<input type="checkbox" name="list" value=".Net" checked="checked"/><label>.Net</label>

<input type="checkbox" name="list" value="RoR"/><label>RoR</label>

<input type="checkbox" name="list" value="PHP"/><label>PHP</label>

如果集合爲MAP

<s:checkboxlist name="map" list="#{1:'瑜珈用品',2:'戶外用品',3:'球類',4:'自行車'}" listKey="key" listValue="value" value="{1,2,3}"/>

生成如下html代碼:

<input type="checkbox" name="map" value="1" checked="checked"/><label>瑜珈用品</label>

<input type="checkbox" name="map" value="2" checked="checked"/><label>戶外用品</label>

<input type="checkbox" name="map" value="3" checked="checked"/><label>球類</label>

<input type="checkbox" name="map" value="4"/><label>自行車</label>

表單標籤_checkboxlist複選框

如果集合裏存放的是javabean

 <%

  Person person1 = new Person(1,"第一個");

  Person person2 = new Person(2,"第二個");

  List<Person> list = new ArrayList<Person>();

  list.add(person1);

  list.add(person2);

  request.setAttribute("persons",list);

  %>

<s:checkboxlist name="beans" list="#request.persons" listKey="personid" listValue="name"/>

Personid和name爲Person的屬性

生成如下html代碼:

<input type="checkbox" name=“beans" value="1"/><label>第一個</label>

<input type="checkbox" name=“beans" value="2"/><label>第二個</label>

表單標籤_radio單選框

該標籤的使用和checkboxlist複選框相同。

如果集合裏存放的是javabean(personid和name爲Person的屬性)

< s:radio name="beans" list="#request.persons" listKey="personid" listValue="name"/>

生成如下html代碼:

<input type="radio" name="beans" id="beans1" value="1"/><label>第一個</label>

<input type="radio" name="beans" id="beans2" value="2"/><label>第二個</label>

如果集合爲MAP

<s:radio name="map" list="#{1:'瑜珈用品',2:'戶外用品',3:'球類',4:'自行車'}" listKey="key" listValue="value“ value="1"/>

生成如下html代碼:

<input type="radio" name="map" id="map1" value="1"/><label for="map1">瑜珈用品</label>

<input type="radio" name="map" id="map2" value="2"/><label for="map2">戶外用品</label>

<input type="radio" name="map" id="map3" value="3"/><label for="map3">球類</label>

<input type="radio" name="map" id="map4" value="4"/><label for="map4">自行車</label>

如果集合爲list

<s:radio name="list" list="{'Java','.Net'}" value="'Java'"/>

生成如下html代碼:

<input type="radio" name="list" checked="checked" value="Java"/><label>Java</label>

<input type="radio" name="list" value=".Net"/><label>.Net</label>

表單標籤_select下拉選擇框

<s:select name="list" list="{'Java','.Net'}" value="'Java'"/>

<select name="list" id="list">

    <option value="Java" selected="selected">Java</option>

    <option value=".Net">.Net</option>

</select>

<s:select name="beans" list="#request.persons" listKey="personid" listValue="name"/>

<select name="beans" id="beans">

    <option value="1">第一個</option>

    <option value="2">第二個</option>

</select>

<s:select name="map" list="#{1:'瑜珈用品',2:'戶外用品',3:'球類',4:'自行車'}" listKey="key" listValue="value" value="1"/>

<select name="map" id="map">

    <option value="1" selected="selected">瑜珈用品</option>

    <option value="2">戶外用品</option>

    <option value="3">球類</option>

    <option value="4">自行車</option>

</select>

 

<s:token />標籤防止重複提交

<s:token />標籤防止重複提交,用法如下:

第一步:在表單中加入<s:token />

<s:form action="helloworld_other" method="post" namespace="/test">

  <s:textfield name="person.name"/><s:token/><s:submit/>

  </s:form>

第二步:

<action name="helloworld_*" class="cn.itcast.action.HelloWorldAction" method="{1}">

       <interceptor-ref name="defaultStack" />

        <interceptor-ref name="token" />

        <result name="invalid.token">/WEB-INF/page/message.jsp</result> 

        <result>/WEB-INF/page/result.jsp</result>       

</action>

以上配置加入了“token”攔截器和“invalid.token”結果,因爲“token”攔截器在會話的token與請求的token不一致時,將會直接返回“invalid.token”結果

在debug狀態,控制檯出現下面信息,是因爲Action中並沒有struts.token和struts.token.name屬性,我們不用關心這個錯誤:

嚴重: ParametersInterceptor - [setParameters]: Unexpected Exception caught setting 'struts.token' on 'class xxx: Error setting expression 'struts.token' with value '[Ljava.lang.String;@39f16f'

嚴重: ParametersInterceptor - [setParameters]: Unexpected Exception caught setting 'struts.token.name'

發佈了4 篇原創文章 · 獲贊 1 · 訪問量 3萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章