`
othella
  • 浏览: 83207 次
  • 性别: Icon_minigender_2
  • 来自: 杭州
社区版块
存档分类
最新评论

ValueStack对象的传送带机制

阅读更多
引用
文章来源:http://www.blogjava.net/nokiaguy/archive/2009/02/11/254314.html
作者:银河使者


众所周知,Strut 2的Action类通过属性可以获得所有相关的值,如请求参数、Action配置参数、向其他Action传递属性值(通过chain结果)等等。要获得这些参数值,我们要做的唯一一件事就是在Action类中声明与参数同名的属性,在Struts 2调用Action类的Action方法(默认是execute方法)之前,就会为相应的Action属性赋值。
    要完成这个功能,有很大程度上,Struts 2要依赖于ValueStack对象。这个对象贯穿整个Action的生命周期(每个Action类的对象实例会拥有一个ValueStack对象)。当 Struts 2接收到一个.action的请求后,会先建立Action类的对象实例,并且将Action类的对象实例压入ValueStack对象中(实际上,ValueStack对于相当一个栈),而ValueStack类的setValue和findValue方法可以设置和获得Action对象的属性值。Struts 2中的某些拦截器正是通过ValueStack类的setValue方法来修改Action类的属性值的。如params拦截器用于将请求参数值映射到相应成Action类的属性值。在params拦截器中在获得请求参数值后,会使用setValue方法设置相应的Action类的属性。
    从这一点可以看出,ValueStack对象就象一个传送带,当客户端请求.action时,Struts 2在创建相应用Action对象后就将Action对象放到了ValueStack传送带上,然后ValueStack传送带会带着Action对象经过若干拦截器,在每一拦截器中都可以通过ValueStack对象设置和获得Action对象中的属性值。实际上,这些拦截器就相当于流水线作业。如果要对 Action对象进行某项加工,再加一个拦截器即可,当不需要进行这项工作时,直接将该拦截器去掉即可。
    下面我们使用一个例子来演示这个过程。在这个例子中实现了一个拦截器,该拦截器的功能是将一个属性文件中的key-value对映射成相应的属性的值。如下面是一个属性文件的内容:

    name = 超人
    price = 10000

    我们可以在Action类中定义name和price属性,在Action中引用这个拦截器后,就会自动为属性赋值。
    在使用该拦截器有如下规则:
    1.  拦截器读取的属性文件路径由path参数指定。
    2.  属性文件的编码格式由encoding参数指定,默认值是UTF-8。
    3.  如果某个key中包含有“.”(该符号不能出现在标识符中),则有如下处理方法:
    (1)将Action类的属性名定义为去掉“.”的key。例如,key为person.name,而属性名可定义为personname。
    (2)将Action类的属性名定义为将“.”替换成其他字符的表示符号。例如,key为person.name,而属性名可定义为person_name,其中“_”由separator参数指定。
    4.  如果key太长,也可以直接使用Action参数进行映射,例如,key为country.person.name,可做如下映射:
      <param name="countrypersonname">name</param>
      要注意的是,name属性值不能包含“.”,因此,应将key值中的“.”去掉。现在就可以直接在Action类中定义名为name的属性的,name属性的值会与key值相同。
    5.  上面所有的规则可以同时使用。

拦截器的源代码:
package interceptors;

import java.util.Enumeration;
import java.util.Map;
import java.util.Properties;
import java.io.InputStream;
import java.io.FileInputStream;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.config.entities.ActionConfig;
import com.opensymphony.xwork2.interceptor.AbstractInterceptor;
import com.opensymphony.xwork2.util.ValueStack;

public class PropertyInterceptor extends AbstractInterceptor
{
    private static final String DEFAULT_PATH_KEY = "path";
    private static final String DEFAULT_ENCODING_KEY = "encoding";
    private static final String DEFAULT_SEPARATOR_KEY = "separator";

    protected String pathKey = DEFAULT_PATH_KEY;
    protected String encodingKey = DEFAULT_ENCODING_KEY;
    protected String separatorKey = DEFAULT_SEPARATOR_KEY;

    public void setPathKey(String pathKey) 
    {
        this.pathKey = pathKey;
    }

    public void setEncodingKey(String encodingKey)
    {
        this.encodingKey = encodingKey;
    }

    public void setSeparatorKey(String separatorKey)
    {
        this.separatorKey = separatorKey;
    }

    @Override
    public String intercept(ActionInvocation invocation) throws Exception
    {
        ActionConfig config = invocation.getProxy().getConfig();

        Map<String, String> parameters = config.getParams();
        if (parameters.containsKey(pathKey))
        {
            String path = parameters.get(pathKey);
            String encoding = parameters.get(encodingKey);
            String separator = parameters.get(separatorKey);
            if (encoding == null)
                encoding = "UTF-8";
            if (separator == null)
                separator = "";
            path = invocation.getAction().getClass().getResource(path)
                    .getPath();
            Properties properties = new Properties();
            InputStream is = new FileInputStream(path);
            java.io.Reader reader = new java.io.InputStreamReader(is, encoding);
            
            properties.load(reader);
            ActionContext ac = invocation.getInvocationContext();
            ValueStack stack = ac.getValueStack();
            System.out.println(stack.hashCode());
            Enumeration names = properties.propertyNames();
            while (names.hasMoreElements())
            {
                //  下面会使用setValue方法修改ValueStack对象中的相应属性值
                String name = names.nextElement().toString();
                if (!name.contains("."))
                    stack.setValue(name, properties.get(name)); 

                String newName = null;
                newName = parameters.get(name.replaceAll("\\.", ""));
                if (newName != null)
                    stack.setValue(newName, properties.get(name));

                if (!separator.equals(""))
                {
                    newName = name.replaceAll("\\.", "");
                    stack.setValue(newName, properties.get(name));
                }               
                newName = name.replaceAll("\\.", separator);
                stack.setValue(newName, properties.get(name));
            } 
        }
        return invocation.invoke();
    }
}


用于测试的Action类的源代码:

package actions;

public class MyAction
{
    private String name;
    private Integer price;
    private String log4jappenderstdout;
    private String log4j_rootLogger;
    private String conversionPattern;

    public String getName()
    {
        return name;
    }

    public void setName(String name)
    {
        this.name = name;
    }

    public Integer getPrice()
    {
        return price;
    }

    public void setPrice(Integer price)
    {
        this.price = price;
    }

    public String getLog4jappenderstdout()
    {
        return log4jappenderstdout;
    }

    public void setLog4jappenderstdout(String log4jappenderstdout)
    {
        this.log4jappenderstdout = log4jappenderstdout;
    }

    public String getLog4j_rootLogger()
    {
        return log4j_rootLogger;
    }

    public void setLog4j_rootLogger(String log4j_rootLogger)
    {
        this.log4j_rootLogger = log4j_rootLogger;
    }

    public String getConversionPattern()
    {
        return conversionPattern;
    }

    public void setConversionPattern(String conversionPattern)
    {
        this.conversionPattern = conversionPattern;
    }

    public String execute()
    {
        System.out.println("name:" + name);
        System.out.println("price:" + price);
        System.out.println("log4jappenderstdout:" + log4jappenderstdout);
        System.out.println("log4j_rootLogger:" + log4j_rootLogger);
        System.out.println("conversionPattern:" + conversionPattern);
        return null;
    }
}


Action类的配置代码如:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
    "-//Apache Software Foundation//DTD Struts Configuration 2.1//EN"
    "http://struts.apache.org/dtds/struts-2.1.dtd">
<struts>
    <package name="struts" extends="struts-default">

        <interceptors>
            <interceptor name="property"
                class="interceptors.PropertyInterceptor" />
            <interceptor-stack name="myStack">
                <interceptor-ref name="defaultStack" />
                <interceptor-ref name="property" />
            </interceptor-stack>
        </interceptors>
        <action name="test" class="actions.MyAction">
            <interceptor-ref name="myStack" />
            <param name="path">/log4j.properties</param>
            <param name="encoding">UTF-8</param>
            <param name="separator">_</param>
            <param name="log4jappenderstdoutlayoutConversionPattern">
                conversionPattern
            </param>

        </action>
    </package>
</struts>


  请将log4j.properties文件复制到WEB-INF\classes目录,并在该文件中加入name和price属性。

测试结果:
name:中国
price:34
log4jappenderstdout:org.apache.log4j.ConsoleAppender
log4j_rootLogger:error, stdout
conversionPattern:%d{ABSOLUTE} %5p %c{1}:%L - %m%n


    由于property拦截器在defaultStack后引用,因此,在该拦截器中设置的属性值是最终结果,如果将property拦截器放在 defaultStack前面(将两个<interceptor-ref>元素掉换一下),就可以通过同名胜Action配置参数或请求参数来干预最终究输出结果了。


分享到:
评论

相关推荐

    ValueStack使用参考

    2. **弹出对象**:`ValueStack.pop()` 移除栈顶对象并返回它,使下一层对象成为新的栈顶对象。 3. **查找对象**:`ValueStack.findValue(String expression)` 使用OGNL表达式查找栈中的对象,返回匹配的结果。 4. ...

    Struts2值栈的理解

    ValueStack 对象在 Struts2 中扮演着非常重要的角色,它将属性值从一个拦截器传到了另一个拦截器(当然,在这其间,属性值可能改变),最后会传到 Action 对象,并将 ValueStack 对象中的属性的值终值赋给 Action 类...

    在线培训:ValueStack

    【在线培训:ValueStack】是一场深入探讨Java...通过深入学习,开发者将能够熟练掌握栈数据结构的运用,理解Struts2框架中的ValueStack机制,并能灵活地调整和操作ValueStack,提高项目开发的效率和代码的可维护性。

    Struts2中关于ValueStack的一些操作

    1、 ValueStack其实就是一个放置Java对象的堆栈而已,唯一特别的是可以使用EL来获得值堆栈中对象属性的数据,并可以为值堆栈的对象属性赋值。 2、 EL,全称Express Language,即表达式语言。不要被语言吓倒,它是...

    s20505_valueStack_OGNL

    1. **ValueStack**:在Struts2中,所有的Action对象都被放入一个称为ValueStack的容器中。当我们在JSP页面上使用OGNL表达式时,实际上是查找ValueStack顶部的对象属性。 2. **ActionContext**:与ValueStack密切相关...

    Struts中的ognl和valueStack

    深入讲解Struts中的ognl和valueStack

    struts2框架面试题及答案02

    在Struts2中,每个Action实例都有一个与之关联的ValueStack对象,该对象贯穿整个Action的生命周期。ValueStack不仅包含了Action对象本身,还可能包含其他辅助对象或数据,比如表单数据、验证错误等。 #### 获取...

    很全面的struts2_ognl总结

    OGNL(Object-Graph Navigation Language)是一种基于 Java 的表达式语言,Struts2 框架中使用 OGNL 来访问和操作 ValueStack 中的对象。在 Struts2 中,OGNL 是一个强大的工具,允许开发者访问和操作 ValueStack 中...

    浅谈Strut2如何对请求参数的封装

    这是通过Struts2的内建机制实现的,其中ValueStack对象起着至关重要的作用。ValueStack是一个数据结构,它像一个栈一样,用于存储Action类的属性和它们的值。当一个HTTP请求到达时,Struts2会创建Action类的一个实例...

    Struts2访问request,session,application的四种方式

    ValueStack是Struts2用于模型驱动的机制,它是一个存储Action属性的栈结构。开发者可以通过`ActionContext.getContext().getValueStack().peek()`获取栈顶的Action对象,然后通过setter/getter方法来读写属性。同样...

    Struts2工作机制

    Struts2中的`ValueStack`是一个对象容器,用于在Action和视图之间传递数据。Action的属性可以直接在视图中通过OGNL(Object-Graph Navigation Language)表达式访问,简化了视图和模型之间的数据交互。 7. **配置...

    Struts2漏洞分析与研究之Ognl机制探讨1

    - OGNL使用`OgnlContext`实现上下文,其中`ValueStack`是根对象的主要实现,它是一个栈结构,用以存储请求中的多个对象。 - `root`(_root):根对象的引用,`values`(_values):存放自定义的Map作为上下文内容...

    传智168期JavaEE struts2杜宏 day32~day33笔记

    综上所述,OGNL是Struts2框架中的一个重要组成部分,它不仅提供了强大的表达式语言功能,还与框架的核心机制(如ValueStack)紧密结合,使得开发者能够更高效地处理对象之间的导航和访问。通过深入理解OGNL的工作...

    Struts2在Action中获得Response对象的四种方法

    OgnlValueStack stack = (OgnlValueStack) request.get("struts.valueStack"); HttpServletResponse response = (HttpServletResponse) stack.findValue("response"); // 使用 response 对象 } } ``` 方法 3:...

    Struts用的ognl和valueStack(vs)实例

    OGNL是一种强大的表达式语言,它允许在对象图中进行导航和操作。在Struts中,OGNL用于在Action对象和视图之间传递数据。它的主要功能包括: 1. **属性访问**:OGNL可以轻松地获取或设置对象的属性,无论是简单类型...

    达内培训机密资料_struts2_day05(1)

    - `ActionInvocation`对象封装了Action调用过程中的所有API,包括获取`ValueStack`对象、执行Action等。 3. **ValueStack对象:** - `ValueStack`对象用于存储Action和拦截器之间的数据交互,可以使用`findValue...

    OGNL表达式语言.txt

    OGNL表达式语言: 相对EL表达式,它提供了平时我们需要的一些功能,如: ...另外OGNL会设定一个根对象(root对象),在Struts2中根对象是ValueStack。 如果访问根对象中的对象的属性,则可以省略#命名空间。

    JavaEE ActionContext存取数据示例

    接着,我们将创建的`User`对象压入ValueStack,然后从ValueStack中找到该对象并将其放入session。在存入session后,我们可以通过OGNL表达式修改session中User对象的属性。 ActionContext还提供了其他有用的方法,如...

    struts2标签和OGNL表达式

    栈顶对象是指在ValueStack(值栈)中位于最顶层的对象,ValueStack是一个特殊的对象列表,它包含了一系列的Action对象,便于OGNL查找和访问属性。 ValueStack是Struts2的核心组件,它是一个OgnlValueStack实例,...

Global site tag (gtag.js) - Google Analytics