第四章 Tablib Struts标签库

    技术2022-05-20  55

    第四章 Tablib Struts标签库

    Struts2 Taglib抽象了不同表示技术,现在Struts2主要支持三种表示技术:JSP,FreeMarkerVelocity。但部分的Tag在三种表示技术下都可以使用,但是也有部分只能在某一种情况下使用。

     

    Tab可以分为两类:通用标签和UI标签。

     

    4.1节 通用标签

    通用标签用来在页面表示的时候控制代码执行的过程,这些标签也允许从Action或者值堆栈中取得数据。例如地域,JavaBeansURLs,和action

     

    控制标签控制程序执行,例如:ifelse,iterator

    数据标签管理数据的取得和创建,例如:beanpush,i18n

     

    控制标签

     

    if标签

    描述

      If标签用来控制基本的条件处理流程,通常和else标签或者elseif标签连用。

     

    参数

    名字

    是否必须

    默认值

    可否使用表达式

    类型

    描述

    id

     

    String

    用来表示该元素,对于UIForm标签来说直接转变为HTML id属性

    test

     

    Boolean

    用来决定是否显示标签内部内容的表达式

     

    例子

    <s:if test="%{false}">

        <div>Will Not Be Executed</div>

    </s:if>

    <s:elseif test="%{true}">

        <div>Will Be Executed</div>

    </s:elseif>

    <s:else>

        <div>Will Not Be Executed</div>

    </s:else>

     

    elseIf 标签

    参考if标签

     

    else 标签

    参考if标签

     

    append标签

    描述

    用来做iterator标签的辅助,将不同iterator中的内容合在一个iterator中。

     

    参数

    名字

    是否必须

    默认值

    可否使用表达式

    类型

    描述

    id

     

    String

    用来保存结果iterator的对象在value context中的名字。

     

    例子

    Action

    public class AppendIteratorTagAction extends ActionSupport {

     

     private List myList1;

     private List myList2;

     private List myList3;

     

     

     public String execute() throws Exception {

     

         myList1 = new ArrayList();

         myList1.add("1");

         myList1.add("2");

         myList1.add("3");

     

         myList2 = new ArrayList();

         myList2.add("a");

         myList2.add("b");

         myList2.add("c");

     

         myList3 = new ArrayList();

         myList3.add("A");

         myList3.add("B");

         myList3.add("C");

     

         return "done";

     }

     

     public List getMyList1() { return myList1; }

     public List getMyList2() { return myList2; }

     public List getMyList3() { return myList3; }

     

    标签使用

    <s:append id="myAppendIterator">

         <s:param value="%{myList1}" />

         <s:param value="%{myList2}" />

         <s:param value="%{myList3}" />

    </s:append>

    <s:iterator value="%{#myAppendIterator}">

         <s:property />

    </s:iterator>

     

    generator 标签(JSP Tag

    描述

    val属性生成一个iterator

     

    参数

    例子

    1:

    生成一个简单的iterator,并且使用iterator标签打印出内容。

    <s:generator val="%{'aaa,bbb,ccc,ddd,eee'}">

     <s:iterator>

         <s:property /><br/>

     </s:iterator>

    </s:generator>

     

    2:

    生成一个iterator,使用count属性。因为count属性值为3,所以只有前三个内容(aaa,bbb,ccc)在生成的iterator中。

    Generate an iterator with count attribute

    <s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" count="3">

     <s:iterator>

         <s:property /><br/>

     </s:iterator>

    </s:generator>

     

     

    3:

    生成iterator,使用了id属性,之后生成的对象放在pageContext中,可以通过指定的id来访问。

    <s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" count="4" separator="," id="myAtt" />

    <%

     Iterator i = (Iterator) pageContext.getAttribute("myAtt");

     while(i.hasNext()) {

         String s = (String) i.next();

    %>

         <%= s %> <br/>

    <%

    %>

     

    4:

    生成iterator,使用converter属性,这里的convertor仅仅将每一个对象添加了一个"converter-"前缀。

    <s:generator val="%{'aaa,bbb,ccc,ddd,eee'}" converter="%{myConverter}">

     <s:iterator>

         <s:property /><br/>

     </s:iterator>

    </s:generator>

     

     

    public class GeneratorTagAction extends ActionSupport {

     

      ....

     

      public Converter getMyConverter() {

         return new Converter() {

             public Object convert(String value) throws Exception {

                 return "converter-"+value;

             }

         };

      }

     

      ...

     

    }

     

    iterator 标签

    描述

    迭代处理一个java.util.Connection或者java.util.Iterator对象

     

    参数

    名字

    是否必须

    默认值

    可否使用表达式

    类型

    描述

    id

     

    String

    Id

    status

    Boolean

    如果指定,在循环的过程中会保留一个IteratorStatus类型的变量,该变量用来查询当前迭代的状态

    value

     

    String

    被迭代的对象

     

     

    例子

    1

    <s:iterator value="days">

      <p>day is: <s:property/></p>

    </s:iterator>

     

    2

    <s:bean name="org.apache.struts2.example.IteratorExample" id="it">

      <s:param name="day" value="'foo'"/>

      <s:param name="day" value="'bar'"/>

    </s:bean>

    <p/>

    <table border="0" cellspacing="0" cellpadding="1">

    <tr>

      <th>Days of the week</th>

    </tr>

    <p/>

    <s:iterator value="#it.days" status="rowstatus">

      <tr>

        <s:if test="#rowstatus.odd == true">

          <td style="background: grey"><s:property/></td>

        </s:if>

        <s:else>

          <td><s:property/></td>

        </s:else>

      </tr>

    </s:iterator>

    </table>

     

    3

    <s:iterator value="groupDao.groups" status="groupStatus">

         <tr

    class="<s:if test="#groupStatus.odd == true ">odd</s:if><s:else>even</s:else>">

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

             <td><s:property value="description" /></td>

             <td>

                 <s:iterator value="users" status="userStatus">

                     <s:property value="fullName" />

    <s:if test="!#userStatus.last">,</s:if>

                 </s:iterator>

             </td>

         </tr>

     </s:iterator>

     

    merge 标签(同append?)

    描述

    参数

    例子

     

    sort 标签(JSP-Tag

    描述

    对一个可以迭代的对象进行排序操作。

     

    参数

     

    名字

    是否必须

    默认值

    可否使用表达式

    类型

    描述

    Comparator

     

    java.util.Comparator

    排序用的比较器

    Source

     

    String

    排序对象

     

    例子

    1:

    <s:sort comparator="myComparator" source="myList">

         <s:iterator>

         <!-- do something with each sorted elements -->

         <s:property value="..." />

         </s:iterator>

    </s:sort>

     

    2:

    <s:sort id="mySortedList" comparator="myComparator" source="myList" />

    <%

       Iterator sortedIterator = (Iterator) pageContext.getAttribute("mySortedList");

       for (Iterator i = sortedIterator; i.hasNext(); ) {

         // do something with each of the sorted elements

       }

    %>

     

    subset

    描述

    递归iterator的一部分

     

    参数

    名字

    是否必须

    默认值

    可否使用表达式

    类型

    描述

    count

    False

     

    Integer

    Iterator中被递归的一部分的item的数量

    Decider

     

    org.apache.struts2.util.

    SubsetIteratorFilter.Decider

    用来判断iterator中的item是否包含在最终的subset内部

    Source

     

    String

    Iterator的对象

    Start

     

    Integer

    开始位置

     

    例子

    Java

     

    public class MySubsetTagAction extends ActionSupport {

         public String execute() throws Exception {

            l = new ArrayList();

            l.add(new Integer(1));

            l.add(new Integer(2));

            l.add(new Integer(3));

            l.add(new Integer(4));

            l.add(new Integer(5));

            return "done";

         }

     

     

         public Integer[] getMyArray() {

            return a;

         }

     

         public List getMyList() {

            return l;

          }

     

         public Decider getMyDecider() {

         return new Decider() {

             public boolean decide(Object element) throws Exception {

                 int i = ((Integer)element).intValue();

                 return (((i % 2) == 0)?true:false);

             }

         };

         }

     }

    <!-- s: List basic -->

       <s:subset source="myList">

          <s:iterator>

             <s:property />

          </s:iterator>

       </s:subset>

    <!-- B: List with count -->

       <s:subset source="myList" count="3">

          <s:iterator>

              <s:property />

          </s:iterator>

        </s:subset>

    <!--  C: List with start -->

         <s:subset source="myList" count="13" start="3">

            <s:iterator>

              <s:property />

            </s:iterator>

         </s:subset>

    <!--  D: List with id -->

         <s:subset id="mySubset" source="myList" count="13" start="3" />

         <%

             Iterator i = (Iterator) pageContext.getAttribute("mySubset");

             while(i.hasNext()) {

         %>

         <%=i.next() %>

         <%  } %>

    <!--  D: List with Decider -->

        <s:subset source="myList" decider="myDecider">

               <s:iterator>

                    <s:property />

               </s:iterator>

        </s:subset>

     

    数据标签

    @TODO 完成如下数据标签

    数据标签包括

    a

    action

    bean

    date

    debug

    i18n

    include

    param

    push

    set

    text

    url

    property

     

    4.2 UI标签

    UI标签主要是指Form相关的标签,UI标签又分为两部分:form标签和构成form内部字段的其他标签。

    每一个UI标签都是基于模板的,即:每一个标签都有一个对应的模板用来生成UI标签的样式,详细内容参看模板节。

     

    所有的UI标签都有着共通的祖先UIBeanUIBean提供了这些UI标签的一系列共通的属性,这些属性可以分为三类:模版相关的属性,JavaScript相关的属性和其他通用属性。

     

    模版相关属性:

    属性

    主题

    数据类型

    说明

    templateDir

    n/a

    String

    定义模版目录

    theme

    n/a

    String

    定义主题的名字

    template

    n/a

    String

    定义模版名字

     

    JavaScript相关属性:

    属性

    主题

    数据类型

    说明

    onclick

    simple

    String

    html javascript onclick 属性

    ondbclick

    simple

    String

    html javascript ondbclick属性

    onmousedown

    simple

    String

    html javascript onmousedown属性

    onmouseup

    simple

    String

    html javascript onmouseup属性

    onmouseover

    simple

    String

    html javascript onmouseover属性

    onmouseout

    simple

    String

    html javascript onmouseout属性

    onfocus

    simple

    String

    html javascript onfocus属性

    onblur

    simple

    String

    html javascript onblur属性

    onkeypress

    simple

    String

    html javascript onkeypress属性

    onkeyup

    simple

    String

    html javascript onkeyup属性

    onkeydown

    simple

    String

    html javascript onkeydown属性

    onselect

    simple

    String

    html javascript onselect属性

    onchange

    simple

    String

    html javascript onchange属性

     

    Tooltip相关属性:

    属性

    数据类型

    默认值

    说明

    tooltip

    String

    none

    为指定的组件设置Tooltip

    jsTooltipEnabled

    String

    false

    使用js表示tooltip

    tooltipIcon

    String

    /struts/static/tooltip/tooltip.gif

    指向tooltip图表的URL

    tooltipDelay

    String

    500

    多长时间后显示Tooltip

    key

    simple

    String

    这个输入字段对应的属性,用来自动设置namelabelvalue

     

    通用属性:

    属性

    主题

    数据类型

    说明

    cssClass

    simple

    String

    定义html class 属性

    cssStyle

    simple

    String

    定义html style属性

    title

    simple

    String

    定义html title属性

    disabled

    simple

    String

    定义html disabled属性

    label

    xhtml

    String

    定义form字段的标签

    labelPosition

    xhtml

    String

    定义标签在Form中的位置,从左从上计算

    requiredPosition

    xhtml

    String

    定义必须的标签在Form中的位置,从左从上计算

    name

    simple

    String

    定义form字段的name映射

    required

    xhtml

    Boolean

    label上添加一个*

    tabIndex

    simple

    String

    定义 html tabIndex属性

    value

    simple

    Object

    定义form字段的值

     

    对于namevalue的说明:

    name用来说明Form字段的名字,和Action类的属性对应。

    value用来记录Form字段的值,和Action类中属性的值对应。

    所以在修改一个字段的内容的时候应该使用如下的标签:

    <s:form action="updateAddress">

        <s:textfield label="Postal Code" name="postalCode" value="%{postalCode}"/>

        ...

    </s:form>

    但是,由于namevalue的关系,struts2标准标签可以自动对应,所以也可以使用如下标签:

    <s:form action="updateAddress">

        <s:textfield label="Postal Code" name="postalCode" />

        ...

    </s:form>

     

    UI标签说明:

    Form部分

    autocompleter

    checkbox

    checkboxlist

    combobox

    datetimepicker

    doubleselect

    head

    file

    form

    hidden

    label

    optiontransferselect

    optgroup

    password

    radio

    reset

    select

    submit

    textarea

    textfield

    token

    updownselect

     

    Form部分:

    actionerror

    actionmessage

    component

    div

    fielderror

    table

    tabbedPanel

    tree

    treenode

     

    4.3节 主题和模板

    概念说明:

    标签(tag):一小段代码,在JSPVelocity或者FreeMarker中执行。程序开发的最小单位,用来生成HTML对应的元素。

    模板(template):一些代码,通常使用FreeMarker写成,可以被某些Tag表示出来(通常是UI Tag)。

    主题(theme):一组模板打包在一起,提供通用功能的模版

     

    主题和模板主要针对可视化的标签(Tag)而言,使用以下例子来说明三者之间的关系。

    假如我们要开发如下的一个画面:

    我们使用如下的代码:

     

    <s:url action="login"  id="loginUrl"></s:url>

    <s:form action="%{loginUrl}">

        <s:textfield label="Name" name="name"/>

        <s:password label="Password" name="password" />

        <s:submit></s:submit>

        <s:reset></s:reset>

    </s:form>

     

    这里<s:form><s:textfield><s:password><s:submit><s:reset>每一个都是一个标签(tag)。

     

    我们在看看这些标签在一起生成的HTML源代码:

    <form id="login" onsubmit="return true;"

        action="/login/login/login.action" method="post">

    <table class="wwFormTable">

        <tr>

           <td class="tdLabel">

    <label for="login_name" class="label">

    Name:

    </label>

    </td>

           <td>

    <input type="text" name="name"

    value="" id="login_name" />

           </td>

        </tr>

     

        <tr>

           <td class="tdLabel">

    <label for="login_password" class="label">

    Password:

    </label></td>

           <td>

    <input type="password"

    name="password" id="login_password" />

           </td>

        </tr>

     

        <tr>

           <td colspan="2">

           <div align="right"><input type="submit" id="login_0"

               value="Submit" /></div>

           </td>

        </tr>

     

        <tr>

           <td colspan="2">

           <div align="right"><input type="reset"

    value="Reset" /></div>

           </td>

        </tr>

     

    </table>

    </form>

     

    在由标签生成HTML代码的时候,例如:

    <s:textfield label="Name" name="name"/>

    生成的代码为:

    <tr>

    <td class="tdLabel">

    <label for="login_name" class="label">

    Name:

    </label>

    </td>

    <td>

    <input type="text" name="name" value="" id="login_name" />

    </td>

    </tr>

    我们可以看到,<s:textfield>标签提供的有效信息只有Namename,而其余的部分,例如<tr><td><label>等代码都根据一个固定的模板文件生成,这个模板文件为:

     

     

     

    标签使我们开发JSP画面的时候使用的最小组件单元,我们根据客户的需要组合各种Tag达到客户的需求。模板是生成这些Tag时候使用的,使用模板可以定义Tag的基本形式,在使用tag的时候,我们只需要指定该Tag的不同属性,即可根据Tag指定的特殊属性,结合模板的基本属性生成可视化的HTML元素。主题是不同tag结合在一起而形成的。

    <input type="text"<#rt/>

     name="${parameters.name?default("")?html}"<#rt/>

    <#if parameters.get("size")?exists>

     size="${parameters.get("size")?html}"<#rt/>

    </#if>

    <#if parameters.maxlength?exists>

     maxlength="${parameters.maxlength?html}"<#rt/>

    </#if>

    <#if parameters.nameValue?exists>

     value="<@s.property value="parameters.nameValue"/>"<#rt/>

    </#if>

    <#if parameters.disabled?default(false)>

     disabled="disabled"<#rt/>

    </#if>

    <#if parameters.readonly?default(false)>

     readonly="readonly"<#rt/>

    </#if>

    <#if parameters.tabindex?exists>

     tabindex="${parameters.tabindex?html}"<#rt/>

    </#if>

    <#if parameters.id?exists>

     id="${parameters.id?html}"<#rt/>

    </#if>

    <#if parameters.cssClass?exists>

     class="${parameters.cssClass?html}"<#rt/>

    </#if>

    <#if parameters.cssStyle?exists>

     style="${parameters.cssStyle?html}"<#rt/>

    </#if>

    <#if parameters.title?exists>

     title="${parameters.title?html}"<#rt/>

    </#if>

    <#include "/${parameters.templateDir}/simple/scripting-events.ftl" />

    <#include "/${parameters.templateDir}/simple/common-attributes.ftl" />

    />

     

    我们考虑标签(Tag)使用模板(Template)生成HTML的过程,根据不同的模板,坑顶可以生成不同的HTML画面,这样我们可以把不同tag的,视觉效果一致的模板放在一起:

    例如:

    <s:form>        TemplateForm_A,       TemplateForm_B

    <s:textfield>  TemplateTextField_A, TemplateTextField_B

    <s:password>   TemplatePassword_A,  TemplatePassword_B

    <s:submit>    TemplateSubmit_A,    TemplateSubmit_B

    <s:reset>       TemplateReset_A,      TemplateReset_B

     

    这样将_A的模板放在一起叫做A主题(Theme),将_B的模板放在一起叫B主题。这样我们在分别使用A主题,B主题的时候可以得到同一个Tag的不同的视觉效果。

     

    模版和主题的概念处在Struts Tag的核心位置。

     

    Struts2默认提供了四种主题:

    Simple    主题:最简单的主题

    XHTML     主题:默认主题,使用常用的HTML技巧

    CSS XHTML主题: 使用CSS实现的XHTML主题

    AJAX      主题:基于XHTML主题,但是同工了AJAX功能

     

    相关配置:

    struts.properties文件中有如下项目:

    struts.ui.theme=xhtml

    struts.ui.templateDir=template

    struts.ui.templateSuffix=ftl

     

    struts.ui.theme的值表示的是使用哪个主题,可选项位:xhtml,simple,css_html,ajax其中xhtml为默认值。

    struts.ui.templateDir的值表示模板的存放目录。

    struts.ui.templateSuffix的值表示模板文件明的后缀,因为Struts2默认使用FreeMarker来编写模板,所以这里我们基本使用ftl。 另外也可以使用vmVelocity)和jspJava Server Page),但是所有的TemplateTheme要我们自己开发。

     

    关于模板文件的存放目录我们需要详细说明,如上述说明,模板文件的存放位置位template,那么系统在那里寻找template目录呢,

    首先,在web应用程序中查找,如果应用程序中存在一个叫做template的目录(跟WEB-INF目录平级),那么所有的文件从这个目录中取得,具体的路径还要加上主题的名字。

    然后,如果在web应用程序中没有找到template目录,那么struts2会在classpath中寻找,由于struts2-core-2.0.9.jar文件中存在template目录,其中内置了四种主题,所以会使用这里变的模板。

    例如:

    如果我们使用了ajax主题,那么会在如下位置超找<s:textfield>的主题

    应用程序

    /template/ajax/textfield.ftl

    classpath

    /template/ajax/textfield.ftl

     

     

    修改或者扩展模板:

    有些时候Struts提供的模板不一定能够满足我们的需求,这时候我们需要修改或者扩展现有模板。重新做新的模板是不明智的,如果是在需要全新的模板,可以考虑基于simple扩展。

     

    修改:

    根据模板的装载机制,可以考虑将模板从struts2-core-2.0.9.jar文件中解压缩到web项目目录,之后修改对应的文件。

     

    包装:

    XHTML提供了一个很好的例子,simple主题提供了基本的功能,XHTML将它包括起来,例如:

    以下是template/xhtml/xxx.ftlxxx表示模板名字)文件内容:

    <#include "/${parameters.templateDir}/xhtml/controlheader.ftl" />

    <#include "/${parameters.templateDir}/simple/xxx.ftl" />

    <#include "/${parameters.templateDir}/xhtml/controlfooter.ftl" />

     

    扩展(extend):

    使用棉线对象的特性可以扩展一个主题,扩展一个主题的时候不需要实现所有的模板,只需要实现需要变化的标签。

    扩展需要在目录中新建一个叫做theme.properties的文件,这个文件只有一行,表明了继承而来的主题的名字,例如:

    /template/ajax/theme.properties文件内容为:

    parent = xhtml

     

     

    4.4 AJAX标签(试验阶段)

    Struts2内置了Dojo 0.4 来提供对Ajax的支持。

    想要使用AJAX标签需要做到两点:

    使用Ajax主题

    JSP画面中使用了head标签配置Ajax属性

     

    AJAX标签主要有:

    <s:div> 

    <s:submit>

    <s:a>

    <s:tabbedPanel>

    <s:autocompleter>

     

    AJAX标签的一些通用属性:

    属性

    说明

    类型

    href

    请求使用的URL

    String

    listenTopic

    使用逗号分割的一组主题列表,这个列表中的主题会导致这个Tag自己内容(DivAutocompleter)重新装载或者执行一个ActionAnchorSubmit

    String

    notifyTopic

    使用逗号分割的一组主题列表,向这个列表中的主题发布一些信息,例如:’data’,’type’,’request’,参看每个标签的详细说明

    String

    showErrorTransportText

    设置是否显示错误消息(默认显示)

    Boolean

    indicator

    请求过程中显示的对象,通常位ProgressBar

    String

     

    Indicator

    <img style="display:none"

    src="${pageContext.request.contextPath}/images/indicator.gif"

    alt="Loading..."/>

     

    Topic

    监听一个Topic

    dojo.event.topic.subscribe("/refresh", function(param1, param2) {

      //this function will be called everytime "/refresh" is published

    });

    向一个Topic发布内容:

    dojo.event.topic.publish("/refresh", "foo", "bar");

     

    URL

    Href属性对应的URL必须使用URL标签定义,例如:

    <s:url id="ajaxTest" value="/AjaxTest.action" />

    <s:div theme="ajax" href="%{ajaxTest}">

      Initial Content

    </s:div>

     

    DIV标签:

    Div主要用来异步的显示数据, PageLoad会出发Div中数据的显示,除非把AutoStart设置为False

    另外,Div的数据显示可以使用Topic来触发。使用listenTopic来定义触发器。

     

    例如:

    <s:url id="ajaxTest" value="/AjaxTest.action" />

    <s:div theme="ajax" href="%{ajaxTest}" listenTopics="/refresh0,/refresh1"/>

    每次想/refresh0,/refresh1发布内容的时候,上面代码定义的div都会刷新。

     

    使用updateFreq可以让Div周期性的触发,在autoStart设置位true的情况下,可以使用delay来延迟首次画面加载的出发时间,例如:

    <s:url id="ajaxTest" value="/AjaxTest.action" />

    <s:div theme="ajax" href="%{ajaxTest}" updateFreq="2000" delay="3000"/>

    上述代码说明,每隔2秒该div触发内容更新一次,但是首次画面加载完成之后3div出发内容更新。

     

    @todo 其他标签

     

     

     

    4.5 OGNL

    OGNLObject Graph Navigation Language的简称,详细相关的信息可以参考:http://www.ognl.org。这里我们只涉及Struts2框架中对OGNL的基本支持。

     

    OGNL是一个对象,属性的查询语言。在OGNL中有一个类型为MapContext(称为上下文),在这个上下文中有一个根元素(root),对根元素的属性的访问可以直接使用属性名字,但是对于其他非根元素属性的访问必须加上特殊符号#

     

    Struts2中上下文为ActionContext,根元素位Value Stack(值堆栈,值堆栈代表了一族对象而不是一个对象,其中Action类的实例也属于值堆栈的一个)。ActionContext中的内容如下图:

                  |

                  |--application

                  |

                  |--session

    context map---|

                  |--value stack(root)

                  |

                  |--request

                  |

                  |--parameters

                  |

                  |--attr (searches page, request, session, then application scopes)

                  |

    因为Action实例被放在Value Stack中,而Value Stack又是根元素(root)中的一个,所以对Action中的属性的访问可以不使用标记#,而对其他的访问都必须使用#标记。

     

    引用Action的属性

    <s:property value="postalCode"/>

    ActionContext中的其他非根(root)元素的属性可以按照如下的方式访问:

    <s:property value="#session.mySessionPropKey"/> or

    <s:property value="#session["mySessionPropKey"]"/> or

    <s:property value="#request["mySessionPropKey"]/>

     

    Action类可以使用ActionContext中的静态方法来访问ActionContext

    ActionContext.getContext().getSession().put("mySessionPropKey", mySessionObject);

     

    OGNLCollectionListsMapsSets

     

    生成List的语法为: {e1,e2,e3}.

    <s:select label="label" name="name"

    list="{'name1','name2','name3'}" value="%{'name2'}" />

    上面的代码生成了一个HTML Select对象,可选的内容为: name1name2name3,默认值为:name2

     

    生成Map的语法为:#{key1:value1,key2:value2}.

    <s:select label="label" name="name"

    list="#{'foo':'foovalue', 'bar':'barvalue'}" />

    上面的代码生成了一个HTML Select对象,foo名字表示的内容为:foovaluebar名字表示的内容为:barvalue

     

    判断一个对象是否在List内存在:

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

       muhahaha

    </s:if>

    <s:else>

       boo

    </s:else>

     

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

       muhahaha

    </s:if>

    <s:else>

       boo

    </s:else>

     

    取得一个List的一部分:

    ?  –  所有满足选择逻辑的对象

    ^  -   第一个满足选择逻辑的对象

    $  -   最后一个满足选择逻辑的对象

    例如:

    person.relatives.{? #this.gender == 'male'}

    上述代码取得这个人(person)所有的男性(this.gender==male)的亲戚(relatives)

     

     

    Lambda 表达式

     

    OGNL支持简单的Lambda表达式语法,使用这些语法可以建立简单的lambda函数。

     

    例如:

    Fibonacci:

    if n==0 return 0;

    elseif n==1 return 1;

    else return fib(n-2)+fib(n-1);

    fib(0) = 0

    fib(1) = 1

    fib(11) = 89

     

    OGNLLambda表达式如何工作呢?

    Lambda表达式必须放在方括号内部,#this表示表达式的参数。例如:

    <s:property value="#fib =:[#this==0 ? 0 : #this==1 ? 1 : #fib(#this-2)+#fib(#this-1)], #fib(11)" />

     

    #fib =:[#this==0 ? 0 : #this==1 ? 1 : #fib(#this-2)+#fib(#this-1)]定义了一个Lambda表达式,

    #fib(11) 调用了这个表达式。

     

    所以上述代码的输出为:89

     

    JSP2.1#被用作了JSP EL(表达式语言)的特殊记好,所以对OGNL的使用可能导致问题,

    一个简单的方法是禁用JSP2.1EL特性,这需要修改web.xml文件:

    <jsp-config>

        <jsp-property-group>

          <url-pattern>*.jsp</url-pattern>

          <el-ignored>true</el-ignored>

        </jsp-property-group>

    </jsp-config>

     

     

    4.6 Tag 语法

    代码示例:

    表达式

    含义

    <p>Username: ${user.username}</p>

    一个在标准上下文中的JavaBean对象,可以适用FreemarkerVelocity,JSTL EL等(不是OGNL)。

    <s:textfield name="username"/>

    Value Stack中的一个username属性。

    <s:url id="es" action="Hello">

      <s:param name="request_locale">

        es

      </s:param>

    </s:url>

    <s:a href="%{es}">Espanol</s:a>

    引用Value Stack中属性的另外一种方法。

    <s:property

      name="#session.user.username" />

    Session中的user对象的username属性。

    <s:select

      label="FooBar" name="foo"

      list="#{'username':'trillian',

        'username':'zaphod'}" />

    一个简单的静态Map,和put("username","trillian")一样


    最新回复(0)