阅读:29635次   评论:29条   更新时间:2011-06-01    
OGNL是XWork引入的一个非常有效的数据处理的工具。我们已经了解了OGNL的基本操作和OGNL的内部结构,接下来,我们来看看XWork对OGNL做了什么样的加强,以及OGNL的体系在Struts2中如何运转。

从例子开始 Top

我们先从一个例子开始,看看数据在Struts2中是如何运转的。

/**
 * @author Downpour
 */
public class User {
	
	private Integer id;
	
	private String name;
	
	private Department department = new Department();
	
        // setter and getters
}

//=========================================================================

/**
 * @author Downpour
 */
public class Department {
	
	private Integer id;
	
	private String name;
		
        // setter and getters
}

//=========================================================================

<form method="post" action="/struts-example/ognl.action">
    user name: <input type="text" name="user.name" value="downpour" />
    department name: <input type="text" name="department.name" value="dev" />
    <input type="submit" value="submit" />
</form>

//=========================================================================

/**
 * @author Downpour
 */
public class OgnlAction extends ActionSupport {

    private static final Log logger = LogFactory.getLog(OgnlAction.class);

    private User user;
    
    private Department department;
    
    /* (non-Javadoc)
     * @see com.opensymphony.xwork2.ActionSupport#execute()
     */
    @Override
    public String execute() throws Exception {
        logger.info("user name:" + user.getName());   // -> downpour
        logger.info("department name:" + department.getName());   // -> dev
        return super.execute();
    }

    // setter and getters
}

//=========================================================================

user name: <s:property value="user.name" />
department name: <s:property value="department.name" />

//=========================================================================


我们可以看到在JSP中,form中的元素input等,都使用OGNL的表达式作为name的值。而在form提交时,这些值都会被设置到Action中的Java对象中。而当Action转向到JSP时,Struts2的Tag又可以从Action的Java对象中,通过OGNL进行取值。

在这里,你看不到任何的OGNL的代码级别操作,因为这些都在Struts2内部进行了封装。而这些封装,都是建立在OGNL的基本概念,也就是根对象和上下文环境之上。下面就分别就这两个方面分别进行讲解。

Struts2中使用OGNL进行计算 Top

取值计算

有了上面的这些知识,我们就能非常容易的理解在Struts2中如何使用OGNL进行取值计算。

提问:在Struts2中,如何使用自身的Tag读取Action中的变量?

Struts2自身的Tag会根据value中的OGNL表达式,在ValueStack中寻找相应的对象。因为action在ValueStack的顶部,所以默认情况下,Struts2的Tag中的OGNL表达式将查找action中的变量。请注意,value中的内容直接是OGNL表达式,无需任何el的标签包装。

例如:<s:property value="user.name" />

提问:在Struts2中,如何使用自身的Tag读取HttpServletRequest,HttpSession中的变量?

在上面的知识中,我们知道,Struts2中OGNL的上下文环境中,包含request,session,application等servlet对象的Map封装。既然这些对象都在OGNL的上下文中,那么根据OGNL的基本知识,我们可以通过在表达式前面加上#符号来对这些变量的值进行访问。

例如:<s:property value="%{#application.myApplicationAttribute}" />
<s:property value="%{#session.mySessionAttribute}" />
<s:property value="%{#request.myRequestAttribute}" />
<s:property value="%{#parameters.myParameter}" />

在这里啰嗦一句,在Tag的value中包括%{开头和}结尾的字符串,不知道Struts2为什么要做出这样的设置,从源码上看,它似乎没有什么特别额外的作用:


if (value == null) {
            value = "top";
        }
        else if (altSyntax()) {
            // the same logic as with findValue(String)
            // if value start with %{ and end with }, just cut it off!
            if (value.startsWith("%{") && value.endsWith("}")) {
                value = value.substring(2, value.length() - 1);
            }
        }

        // exception: don't call findString(), since we don't want the
        //            expression parsed in this one case. it really
        //            doesn't make sense, in fact.
        actualValue = (String) getStack().findValue(value, String.class);
        
        ......

}


有兴趣的朋友可以研究一下,这一对符号的原理究竟是什么。

提问:在Struts2中,如何使用JSTL来读取Action中的变量?

这是一个历史悠久的问题。因为事实上,很多朋友(包括我在内)是不使用Struts2自身的标签库,而是使用JSTL的,可能因为JSTL标签库比较少,简单易用的原因吧。

我们知道,JSTL默认是从page,request,session,application这四个Scope逐次查找相应的EL表达式所对应的对象的值。那么如果要使用JSTL来读取Action中的变量,就需要把Action中的变量,放到request域中才行。所以,早在Webwork2.1.X的年代,我们会编写一个拦截器来做这个事情的。大致的原理是:在Action执行完返回之前,依次读取Action中的所有的变量,并依次调用request.setAttribute()来进行设置。具体的整合方式,请参考以下这篇文档:http://wiki.opensymphony.com/display/WW/Using+WebWork+and+XWork+with+JSP+2.0+and+JSTL+1.1

不过随着时代的发展,上面的这种方式,已经不再被推荐使用了。(虽然如此,我们依然可以学习它的一个解决问题的思路)目前来说,自从Webwork2.2以后,包括Struts2,都使用另外一种整合方式:对HttpServletRequest进行装饰。让我们来看一下源码:

public class StrutsRequestWrapper extends HttpServletRequestWrapper {

    /**
     * The constructor
     * @param req The request
     */
    public StrutsRequestWrapper(HttpServletRequest req) {
        super(req);
    }

    /**
     * Gets the object, looking in the value stack if not found
     *
     * @param s The attribute key
     */
    public Object getAttribute(String s) {
        if (s != null && s.startsWith("javax.servlet")) {
            // don't bother with the standard javax.servlet attributes, we can short-circuit this
            // see WW-953 and the forums post linked in that issue for more info
            return super.getAttribute(s);
        }

        ActionContext ctx = ActionContext.getContext();
        Object attribute = super.getAttribute(s);

        boolean alreadyIn = false;
        Boolean b = (Boolean) ctx.get("__requestWrapper.getAttribute");
        if (b != null) {
            alreadyIn = b.booleanValue();
        }

        // note: we don't let # come through or else a request for
        // #attr.foo or #request.foo could cause an endless loop
        if (!alreadyIn && attribute == null && s.indexOf("#") == -1) {
            try {
                // If not found, then try the ValueStack
                ctx.put("__requestWrapper.getAttribute", Boolean.TRUE);
                ValueStack stack = ctx.getValueStack();
                if (stack != null) {
                    attribute = stack.findValue(s);
                }
            } finally {
                ctx.put("__requestWrapper.getAttribute", Boolean.FALSE);
            }
        }
        return attribute;
    }
}


看到了嘛?这个类会在Struts2初始化的时候,替换HttpServletRequest,运行于整个Struts2的运行过程中,当我们试图调用request.getAttribute()的时候,就会执行上面的这个方法。(这是一个典型的装饰器模式)在执行上面的方法时,会首先调用HttpServletRequest中原本的request.getAttribute(),如果没有找到,它会继续到ValueStack中去查找,而action在ValueStack中,所以action中的变量通过OGNL表达式,就能找到对应的值了。

在这里,在el表达式广泛使用的今天,JSTL1.1以后,也支持直接使用el表达式。注意与直接使用struts2的tag的区别,这里需要使用el的表示符号:${}

例如:${user.name}, <c:out value="${department.name}" />

提问:在Struts2中,如何使用Freemarker等模板来读取Action中的变量以及HttpServletRequest和HttpSession中的变量?

Freemarker等模板在Struts2中有对应的Result,而在这些Result中,Freemarker等模板会根据ValueStack和ActionContext中的内容,构造这些模板可识别的Model,从而使得模板可以以他们各自的语法对ValueStack和ActionContext中的内容进行读取。

有关Freemarker对于变量的读取,可以参考Struts2的官方文档,非常详细:http://struts.apache.org/2.0.14/docs/freemarker.html

设值计算

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

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

2. 在请求进入Action代码前,通过某种通用的机制,搜集页面上传递过来的参数,并调用OGNL相关的代码,对Action进行设值。

上面的第一个步骤,在处理URL请求时完成,而第二个步骤,则涉及到另外一个XWork的核心知识:拦截器。所以有关Struts2使用OGNL进行设值计算的详细分析,将会在拦截器章节具体给出。
评论 共 29 条 请登录后发表评论
29 楼 游其是你 2011-12-15 16:53
谢谢啦,解了我为什么从action中取值不需要加“#”的疑惑,原来action中的变量是在值栈中的!
28 楼 gaokuitai 2011-12-13 22:35
楼主很强大,说的很彻底!!!   
27 楼 niaotuo 2011-07-12 14:25
才疏学浅啊,没太看懂。。。。。
26 楼 361010911 2011-05-13 09:56
 
25 楼 wenjinglian 2011-04-05 21:52
分析的很细。。。
24 楼 jassize 2011-03-12 12:00
辛苦了~。。
23 楼 xiexiaolong 2011-01-07 16:45
分析很棒
22 楼 lei715880100 2010-09-11 14:54
厉害,分析的很透彻。
不过对我这种菜鸟来说,看起来有点困难。看起来需要多啃几遍了,呵呵
21 楼 zyxbb177 2010-07-02 10:54
写的很不错,但小弟水平有限只能理解部分。。顶
20 楼 lixia0417 2010-06-18 22:20
只能说一个字,强
19 楼 wudiju 2010-05-17 13:18
真是精彩
18 楼 oo-java 2010-02-07 00:13
楼主的文章,比起一些书质量那可是高了一截啊
17 楼 抢街饭 2010-01-11 08:44
楼主辛苦了
16 楼 suntine 2009-07-21 10:59
拜读过文,谢谢啦
15 楼 edgar615 2009-07-20 14:42
相当不错,作为一个初学者,拿着你的分析,再参照struts的参考文档,受益匪浅
继续拜读
14 楼 only_java 2009-03-16 16:15
知其所以然
相当不错!
13 楼 jocund 2009-03-09 09:33
太有才了!值得收藏。
12 楼 baiyangshu20081104 2009-02-03 18:23
楼主的文章越看越过瘾!加油!
11 楼 downpour 2009-01-07 13:15
url的例子比较特殊,因为在url中,是允许有#符号存在的。所以在设值的时候,如果碰到#这样的符号,是会被忽略的。

有空可以试试property标签,貌似从代码上看,用不用%{}好像没什么很大区别。

不过我是不用Struts2的标签的,乱七八糟需要学习的语法太多,总不至于常备一份Tag的教程在边上看吧。
10 楼 kyo100900 2009-01-07 11:28
我做个了小试验:

<a href="<s:url value="http://www.javaeye.com" />">javaeye</a>


没有问题

<s:set name="url" value="'http://www.javaeye.com'"/>
<a href="<s:url value="#url" />">javaeye</a>


这样却不行

一定要改成

<s:set name="url" value="'http://www.javaeye.com'"/>
<a href="<s:url value="%{#url}" />">javaeye</a>

9 楼 downpour 2009-01-07 10:12
kyo100900 写道

我看过webwork in action,谈到 %{..}是webwork2.2后来提供的最语法,支持对%{..}中的内容进行求值。


这个从源码上看的确是这样,但是为什么要这个符号呢?不是多此一举嘛?

kyo100900 写道
将你这个系列的文章放在struts2圈子中,行吗?呵呵


可以,没问题。
8 楼 kyo100900 2009-01-07 09:57
分析的太棒了。

关于

引用
在这里啰嗦一句,在Tag的value中包括%{开头和}结尾的字符串,不知道Struts2为什么要做出这样的设置,从源码上看,它似乎没有什么特别额外的作用:


我看过webwork in action,谈到 %{..}是webwork2.2后来提供的最语法,支持对%{..}中的内容进行求值。


最后,将你这个系列的文章放在struts2圈子中,行吗?呵呵
7 楼 jccg17476 2009-01-06 12:22
downpour 写道

jccg17476 写道很感谢楼主,有个小小的建议,能不能把您的文章弄成PDF格式的,提供下载呀!等到所有的专栏文章结束后,我会使用Javaeye的电子书功能做成PDF格式,提供下载。

我有个朋友也在学struts2,不过他们那边不能上外网,我只能通过mail给他发资料,如果有PDF的,会方便一点。
6 楼 downpour 2009-01-05 20:10
rain2005 写道

弱弱的问上一句,downpour发表文章的频率是多少啊?一直很期待。


空下来我就会写一些,不过不能保证固定频率的。

主要原因在于,很多的东西是需要整理的,包括翻阅源码、查阅Struts2自带的Reference、查询Javaeye的老帖子等等,这些都需要时间。有很多知识,我也只是知晓,并不是非常精通,所以有很多东西我也需要重新学习,才能够组织语言成为文章。

所以请大家多多包涵,也请指出其中错误的地方,我好及时修改。
5 楼 rain2005 2009-01-05 19:45
弱弱的问上一句,downpour发表文章的频率是多少啊?一直很期待。
4 楼 chrrity 2009-01-05 17:22
继续期待大作,加油...
3 楼 downpour 2009-01-05 16:55
jccg17476 写道

很感谢楼主,有个小小的建议,能不能把您的文章弄成PDF格式的,提供下载呀!


等到所有的专栏文章结束后,我会使用Javaeye的电子书功能做成PDF格式,提供下载。
2 楼 jccg17476 2009-01-05 16:45
很感谢楼主,有个小小的建议,能不能把您的文章弄成PDF格式的,提供下载呀!
1 楼 coolworm 2009-01-05 14:36
楼主太强了, 分析的这么透彻,您的文章我一直在关注中,加油!

发表评论

您还没有登录,请您登录后再发表评论

文章信息

  • downpour在2009-01-05创建
  • downpour在2011-06-01更新
  • 标签: struts2 ognl valuestack
Global site tag (gtag.js) - Google Analytics