`
aaagu1234
  • 浏览: 144945 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Struts 2杂谈(1):ValueStack对象的传送带机制

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

  要完成这个功能,有很大程度上,Struts 2要依赖于ValueStack对象。这个对象贯穿整个Action的生命周期(每个Action类的对象实例会拥有一个ValueStack对象)。当 Struts 2接收到一个.action的请求后,会先建立Action类的对象实例,但并不会调用Action方法,而是先将Action类的相应属性放到 ValueStack对象的顶层节点(ValueStack对象相当于一个栈)。只是所有的属性值都是默认的值,如String类型的属性值为 null,int类型的属性值为0等。

  在处理完上述工作后,Struts 2就会调用拦截器链中的拦截器,当调用完所有的拦截器后,最后会调用Action类的Action方法,在调用Action方法之前,会将 ValueStack对象顶层节点中的属性值赋给Action类中相应的属性。大家要注意,在这里就给我们带来了很大的灵活性。也就是说,在Struts 2调用拦截器的过程中,可以改变ValueStack对象中属性的值,当改变某个属性值后,Action类的相应属性值就会变成在拦截器中最后改变该属性的这个值。

  从上面的描述很容易知道,在Struts 2的的Action类可以获得与属性同名的参数值就是通过不同的拦截器来处理的,如获得请求参数的拦截器是params,获得Action的配置参数的拦截器是staticParams等。在这些拦截器内部读取相应的值,并更新ValueStack对象顶层节点的相应属性的值。而ValueStack对象就象一个传送带,将属性值从一个拦截器传到了另一个拦截器(当然,在这其间,属性值可能改变),最后会传到Action对象,并将ValueStack对象中的属性的值终值赋给Action类的相应属性。
 也许有的读者会看出来一个问题,如果有多个拦截器都改变同一个属性值,那么在后面引用的拦截器将覆盖之前引用的拦截器改变的属性值。由于在 defaultStack拦截器栈中staticParams是在params之前引用的,因此,如果某个请求参数与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. 上面所有的规则可以同时使用。

  拦截器的源代码:
packageinterceptors;
importjava.util.Enumeration;
importjava.util.Map;
importjava.util.Properties;
importjava.io.InputStream;
importjava.io.FileInputStream;
importcom.opensymphony.xwork2.ActionContext;
importcom.opensymphony.xwork2.ActionInvocation;
importcom.opensymphony.xwork2.config.entities.ActionConfig;
importcom.opensymphony.xwork2.interceptor.AbstractInterceptor;
importcom.opensymphony.xwork2.util.ValueStack;
public class PropertyInterceptor extends AbstractInterceptor
{
  private static final StringDEFAULT_PATH_KEY="path";
  privatestaticfinalStringDEFAULT_ENCODING_KEY="encoding";
  privatestaticfinalStringDEFAULT_SEPARATOR_KEY="separator";
  protectedStringpathKey=DEFAULT_PATH_KEY;
  protectedStringencodingKey=DEFAULT_ENCODING_KEY;
  protectedStringseparatorKey=DEFAULT_SEPARATOR_KEY;
  publicvoidsetPathKey(StringpathKey)
  {
    this.pathKey=pathKey;
  }
  publicvoidsetEncodingKey(StringencodingKey)
  {
    this.encodingKey=encodingKey;
  }
  publicvoidsetSeparatorKey(StringseparatorKey)
  {
    this.separatorKey=separatorKey;
  }
  @Override
  public String intercept(ActionInvocationinvocation)throwsException
  {
    ActionConfigconfig=invocation.getProxy().getConfig();
    Map<String,String>parameters=config.getParams();
    if(parameters.containsKey(pathKey))
    {
      Stringpath=parameters.get(pathKey);
      Stringencoding=parameters.get(encodingKey);
      Stringseparator=parameters.get(separatorKey);
      if(encoding==null)
        encoding="UTF-8";
      if(separator==null)
        separator="";
      path=invocation.getAction().getClass().getResource(path)
          .getPath();
      Propertiesproperties=newProperties();
      InputStreamis=newFileInputStream(path);
      java.io.Readerreader=newjava.io.InputStreamReader(is,encoding);
      
      properties.load(reader);
      ActionContextac=invocation.getInvocationContext();
      ValueStackstack=ac.getValueStack();
      System.out.println(stack.hashCode());
      Enumerationnames=properties.propertyNames();
      while(names.hasMoreElements())
      {
        // 下面会使用setValue方法修改ValueStack对象中的相应属性值
        Stringname=names.nextElement().toString();
        if(!name.contains("."))
          stack.setValue(name,properties.get(name));
        StringnewName=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));
      }
    }
    returninvocation.invoke();
  }
}


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

packageactions;
public class MyAction
{
  privateStringname;
  privateIntegerprice;
  privateStringlog4jappenderstdout;
  privateStringlog4j_rootLogger;
  privateStringconversionPattern;
  public String getName()
  {
    returnname;
  }
  publicvoidsetName(Stringname)
  {
    this.name=name;
  }
  publicIntegergetPrice()
  {
    returnprice;
  }
  publicvoidsetPrice(Integerprice)
  {
    this.price=price;
  }
  publicStringgetLog4jappenderstdout()
  {
    returnlog4jappenderstdout;
  }
  publicvoidsetLog4jappenderstdout(Stringlog4jappenderstdout)
  {
    this.log4jappenderstdout=log4jappenderstdout;
  }
  publicStringgetLog4j_rootLogger()
  {
    returnlog4j_rootLogger;
  }
  publicvoidsetLog4j_rootLogger(Stringlog4j_rootLogger)
  {
    this.log4j_rootLogger=log4j_rootLogger;
  }
  publicStringgetConversionPattern()
  {
    returnconversionPattern;
  }
  publicvoidsetConversionPattern(StringconversionPattern)
  {
    this.conversionPattern=conversionPattern;
  }
  publicStringexecute()
  {
    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);
    returnnull;
  }
}


Action类的配置代码如:

<?xmlversion="1.0"encoding="UTF-8"?>
<!DOCTYPEstrutsPUBLIC
  "-//ApacheSoftwareFoundation//DTDStrutsConfiguration2.1//EN"
  "http://struts.apache.org/dtds/struts-2.1.dtd">
<struts>
  <packagename="struts"extends="struts-default">
    <interceptors>
      <interceptorname="property"
        class="interceptors.PropertyInterceptor"/>
      <interceptor-stackname="myStack">
        <interceptor-refname="defaultStack"/>
        <interceptor-refname="property"/>
      </interceptor-stack>
    </interceptors>
    <actionname="test"class="actions.MyAction">
      <interceptor-refname="myStack"/>
      <paramname="path">/log4j.properties</param>
      <paramname="encoding">UTF-8</param>
      <paramname="separator">_</param>
      <paramname="log4jappenderstdoutlayoutConversionPattern">
        conversionPattern
      </param>
    </action>
  </package>
</struts>


  请将log4j.properties文件复制到WEB-INFclasses目录,并在该文件中加入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

    培训主要学习以下几个知识点: 1、制作PPT,讲解栈的特点和使用方法 2、编码实现回文对 3、制作PPT,讲解Struts2中的ValueStack 4、编码改变ValueStack中的对象的顺序

    Struts2中关于ValueStack的一些操作

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

    Struts中的ognl和valueStack

    深入讲解Struts中的ognl和valueStack

    Struts用的ognl和valueStack(vs)实例

    Struts用的ognl和valueStack(vs)实例

    Struts2中的参数传递

    我们知道,Struts2完成参数传递处理工作的基础是OGNL和ValueStack。而在这个 过程中,我也把Struts2所要做的工作大致归纳为两个方面: 1. 对OGNL操作进行封装,完成OGNL表达式所表示的值到Java对象的值传递机制 2. ...

    Struts2入门教程(全新完整版)

    10.为什么要使用struts2代替struts1.x 7 二、struts.xml配置及例程 7 1.配置文件的优先级 7 2.配置形式 8 3.package配置相关 8 4.分工合作include:指定多个配置文件 10 5.tomcat认证访问 10 6.初识拦截器 11 7....

    马士兵Struts2笔记2013

    建立一个Struts2 工程,用Action的属性接收参数,使用Domain Model (实体模型) 接收参数,Struts2_2.1.6版本的中文问题,Struts模块包含,Struts简单数据验证 ,Struts ValueStack(值栈) Debug,Value Stack ...

    Struts2 in action中文版

    第1章 Struts 2:现代Web框架 2 1.1 Web应用程序:快速学习 2 1.1.1 构建Web应用程序 2 1.1.2 基础技术简介 3 1.1.3 深入研究 6 1.2 Web应用程序框架 7 1.2.1 什么是框架 7 1.2.2 为什么使用框架 8 1.3 Struts 2框架...

    struts2 标签库使用文档

    Struts2标签库的组成 Struts2框架的标签库可以分为以下三类: 用户界面标签(UI标签):主要用来生成HTML元素的标签。 表单标签:主要用于生成HTML页面的FORM元素,以及普通表单元素的标签。 非表单标签:主要用于生成...

    struts2 学习例子

    Struts2标签库提供了主题、模板支持,极大地简化了视图页面的编写,而且,struts2的主题、模板都提供了很好的扩展性。实现了更好的代码复用。Struts2允许在页面中使用自定义组件,这完全能满足项目中页面显示复杂,...

    struts2模拟

    struts2模拟模拟工具,可以实现action访问,我表单属性的自动封装。提供了页面显示和跌带器,利用struts2值栈(valueStack)和对象栈map栈的思想。可以用来了解struts2的执行过程,纯属个人学习

    Struts2练习Demo以及随笔

    Struts的开发步骤、OGNL、ValueStack、Action核心、Result基本原理、Struts2核心标记库、Struts2拦截器、自定义拦截器、UI标记、非UI标记、资源文件国际化等等实例全面使用。

    struts学习笔记(3)

    当客户端向action发送请求并且最后跳转到另外一个页面的时候,在跳转的同时,struts2框架会帮我们自动把需要传到页面的值放这两个对象当中去,然后我们在页面就可以使用固定的方式把值从这个两个对象里面取出来了。...

    linjin101#javaStudy#Struts2中的OGNL和值栈ValueStack1

    1、什么是值栈 2、值栈的内部结构 3、ActionContext和ValueStatck的关系 4、如何获取值栈对象 5、向值栈存数据 6、从值栈中获取数据

    Struts2基础教程

    讲解Struts2入门基础,包括Action、Result、ValueStack等,Struts2初学者会有帮助

    Struts2 Value Stack Contents 中取值、多个集合数组

    Struts2 Value Stack Contents 中取值、多个集合数组示例

    OGNL表达归纳

    3.由于ValueStack(值栈)是Struts 2中OGNL的根对象,如果用户需要访问值栈中的对象,在JSP页面可以直接通过下面的EL表达式访问ValueStack(值栈)中对象的属性 4.为何使用EL表达式能够访问valueStack中对象的属性 原因...

    Struts2框架基础 二

    OGNL表达式 interceptor拦截器 valuestack的存入 /取出

    ValueStack使用参考

    ValueStack的使用方法和一些技能

Global site tag (gtag.js) - Google Analytics