浅析Struts2中的OGNL和ValueStack

要了解Struts2与OGNL表达式的关系,我们必须先搞清楚以下三个概念:

1、  ActionContext它是Action运行的上下文环境,Action的多项设置都存放在次,我们每一次Action调用都会创建一个ActionContext。通常情况下我们可以通过静态方法getContext()来获得Action上下文,进而进行其它操作,比如说可以得到request、session、application。

2、  ValueStack此对象主要是由OGNL框架实现,具体的实现类是com.opensymphony.xwork2.util.ValueStack;它主要包含了一个Map类型的Context对象,和最重要的getValue与findValue方法,以及peek、pop等栈所特有的方法。

Strut 2的Action类通过属性可以获得所有相关的值,如请求参数、Action配置参数、向其他Action传递属性值(通过chain结果)等等。要获得这些参数值,我们要做的唯一一件事就是在Action类中声明与参数同名的属性,在Struts2调用Action类的Action方法(默认是execute方法)之前,就会为相应的Action属性赋值。

要完成这个功能,有很大程度上,Struts 2要依赖于ValueStack对象。这个对象贯穿整个Action的生命周期(每个Action类的对象实例会拥有一个ValueStack对象)。当Struts2接收到一个.action的请求后,会先建立Action类的对象实例,并且将Action类的对象实例压入ValueStack对象中(实际上,ValueStack对象相当于一个栈),而ValueStack类的setValue和findValue方法可以设置和获得Action对象的属性值。Struts2中的某些拦截器正是通过ValueStack类的setValue方法来修改Action类的属性值的。如params拦截器用于将请求参数值映射到相对应的Action类的属性值。在params拦截器中在获得请求参数值后,会使用setValue方法设置相应的Action类的属性。

从这一点可以看出,ValueStack对象就象一个传送带,当客户端请求 .action时,Struts2在创建相对应Action对象后就将Action对象放到了ValueStack传送带上,然后ValueStack传送带会带着Action对象经过若干个拦截器,在每一拦截器中都可以通过ValueStack对象设置和获得Action对象中的属性值。实际上,这些拦截器就相当于流水线作业。如果要对Action对象进行某项加工,再加一个拦截器即可,当不需要进行这项工作时,直接将该拦截器去掉即可。

需要注意的是我们访问这类对象时是不需要加入#的,因为它是根对象,所以不能加#,加了以后就不能访问到了。比如我们在Action中做了如下操作ActionContext.getContext().getVlaueStack().setVlaue(“MM”,”这是信息”);然后在页面就可以通过<s:property value=”MM” />就可以了。

3、  Stack Context(map):stack上下文,它包含一系列对象,包括:request、session、attr、application map等,也是用来存值的。Struts2对OGNL上下文的概念又做了进一步扩充,在struts2中,OGNL上下文通常如下所示:

|--request

|--application

context map---|--OgnlValueStack(root) [ user、action、OgnlUtil、... ]

|--session

|--attr

|--parameters

在Struts2中,采用标准命名的上下文(Context)来处理OGNL表达式。处理OGNL的顶级对象是一个Map(也叫context map,它实际上是ActionContext中的一个对象,也叫context),在这个contxt属性中我们分别放置了

context.put(“常量REQUEST”,new HashMap<String,Object>());

context.put(“常量APPLICATION”,new HashMap<String,Object>());

等等,同样我们在其中也放置了一个名叫vsMap的对象(自己随便取了个名字);来看下面两个相关的方法

public void setValueStack(ValueStack stack) {  put(VALUE_STACK, stack);   }

public void put(String key, Object value) {          context.put(key, value);   }

我们可以看到这个ValueStack同样设置到了上下文对象中[实际上使用的ValueStack中的context属性]。 而ValueStack中除了这个Map以外,还维护一个ComponentRoot对象,它实际上用来存放我们的一个List集合,它还要维护一个栈即Stack, 以期让我们进行后续的操作。

Struts2框架把我们的ActionContext设置为OGNL 的上下文环境,凡是此环境中的值我们都应该通过#key的方式来进行访问,所以request,session等需要加前缀,又因为Struts2将我们的ValueStack作为OGNL的根对象,所以我们访问其中的内容只能通过非#的方式来进行。Action的实例,总是放到value stack中。因为Action放在stack中,而stack是root(根对象),所以对Action中的属性的访问就可以省略#标记。但是,要访问ActionContext中其它对象的属性,就必须要带上#标记,以便让OGNL知道,不是从根对象,而是从其它对象中去寻找。具体的代码可以这样理解:

而我们直接通过ActionContet.getContext().put(“属性名”,“值”);放入的数据放在了一个ActionContext 的context对象(OGNL上下文)和ValueStack对象的context对象共同引用的一个Map对象中,所以我们既可以通过#key也可以直接通过key来进行访问。那么访问Action中的属性的代码就可以这样写:

<s:property value="postalCode"/>其它ActionContext中的非根对象属性的访问要像下面这样写:

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

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

<s:property value="#request['myRequestPropKey']"/>对Collection的处理,内容就很简单。

<s:select label="label" name="name" list="{'name1','name2','name3'}" value="%{'name2'}" />这是处理List。这个代码在页面上建立一个下拉选项,内容是list中的内容,默认值是name2。

处理map

<s:select label="label" name="name" list="#{'foo':'foovalue', 'bar':'barvalue'}" />

需要注意的是,判断一个值是否在collection中。我们要使用in或者not in来处理。

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

muhahaha

</s:if>

<s:else>

boo

</s:else>

另外,可以使用通配符来选择collection对象的子集。

?——所有匹配选择逻辑的元素

^——只提取符合选择逻辑的第一个元素

$——只提取符合选择逻辑的最后一个元素

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

设值计算

Struts2中使用OGNL进行设值计算,就是指View层传递数据到Control层,并且能够设置到相应的Java对象中。这个过程从逻辑上说需要分成两步来完成:

1、  对于每个请求,都建立一个与相应Action对应的ActionContext作为OGNL的上下文环境和ValueStack,并且把Action压入ValueStack

2、  在请求进入Action代码前,通过某种通用的机制,搜集页面上传递过来的参数,并调用OGNL相关的代码,对Action进行设值。上面的第一个步骤,在处理URL请求时完成,而第二个步骤由struts2内置的拦截器完成。

3、  “#”主要有三种用途:

  1. 访问OGNL上下文和Action上下文,# 相当于ActionContext.getContext();下表有几个ActionContext中有用的属性:

a)         名称

b)         作用

c)         例子

  1. parameters包含当前HTTP请求参数的Map,#parameters.id[0]作用相当于request.getParameter("id")
  2. reques包含当前HttpServletRequest的属性(attribute)的Map,#request.userName相当于request.getAttribute("userName")
  3. session包含当前HttpSession的属性(attribute)的Map,#session.userName相当于session.getAttribute("userName")
  4. application包含当前应用的ServletContext的属性(attribute)的Map, #application.userName相当于application.getAttribute("userName")

4、  attr用于按request > session > application顺序访问其属性(attribute), #attr.userName相当于按顺序在以上三个范围(scope)内读取userName属性,直到找到为止

a)         用于过滤和投影(projecting)集合,如books.{?#this.price<100};

b)         构造Map,如#{'foo1':'bar1', 'foo2':'bar2'}。

5、  “%”符号的用途是在标志的属性为字符串类型时,计算OGNL表达式的值。[既字符串不是输出到页面,而是作为某个属性的取值],如下面的url就是一个取值。

例如在Ognl.jsp中加入以下代码:

<h3>%的用途</h3>

    <p><s:url value="#foobar['foo1']" /></p>

<p><s:url value="%{#foobar['foo1']}" /></p>

6、  $符号主要有两个方面的用途。

a)         在国际化资源文件中,引用OGNL表达式,例如国际化资源文件中的代码:reg.agerange=国际化资源信息 : 年龄必须在${min}同${max}之间。

b)         在Struts 2框架的配置文件中引用OGNL表达式,例如下面的代码片断所示:

        <validators>

        <field name=”intb”>

            <field-validator type=”int”>

                <param name=”min”>10</param>

                <param name=”max”>100</param>

                <message>BAction-test校验:数字必须为${min}为${max}之间!</message>

            </field-validator>

        </field>

        </validators>

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏小灰灰

SpringMVC支持跨域的两种姿势

SpringMVC支持跨域的几种姿势 跨域好像是一个前端的问题,通常是a域名下向b域名的服务发起请求,然后处于浏览器的安全原则,被拦截了,而这种场景,在实际的...

1956
来自专栏微信公众号:Java团长

我们为什么要使用AOP?

之前写了一篇文章Spring3:AOP,是当时学习如何使用Spring AOP的时候写的,比较基础。这篇文章最后的推荐以及回复认为我写的对大家有帮助的评论有很多...

512
来自专栏noteless

[二十四]JavaIO之PrintWriter

他与PrintStream的逻辑上功能目的是相同的--他们都想做同一件事情--更便捷的格式化打印输出

432
来自专栏积累沉淀

Java IO结构各种流详解

花了两天时间研究了一下Java IO的流,对于各种流,加深了一下理解 首先看我做的思维导图 ? 文件流 public class FileIO { ...

3668
来自专栏Java 源码分析

Spring笔记(一)

1.基本介绍 spring 是一个一站式框架,也就是有了它 web 层,service 层还有 dao 层都能直接搞定而不需使用其他的框架。 这三层分别就是: ...

3245
来自专栏李家的小酒馆

struts2面试整理

struts2的工作原理 客户端发送请求 经过一系列的过滤器 FilterDispatcher通过ActionMapper来决定这个REquest需要调用的Ac...

1800
来自专栏日常分享

Spring AOP的一个简单实现

首先配置XML:service采用和之前一样的代码,只是没有通过实现接口来实现,而是直接一个实现类。transactionManager依旧为之前的事务管理器。

611
来自专栏青枫的专栏

day25_day27_Struts2_学习回顾

        表现层、MVC模式。 2、Struts1和Struts2的一个显著区别是什么?     答:

825
来自专栏闵开慧

接口与abstract class区别

1、抽象类可以包含部分方法的实现,这是抽象类优于接口的一个主要地方。 2、由于Java的单继承,每个类只能从一个抽象类继承,但是每个类可以实现多个接口,使用接...

3169
来自专栏Java 源码分析

Struts2笔记(二)

1.结果页面 1.全局结果页面 当一些方法的返回值一样的时候并且需要跳转的页面也一样的时候就可以使用全局结果页面。 <package name="struts...

2393

扫码关注云+社区