众所周知,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,可做如下映射:
        <paramname="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>  <?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配置参数或请求参数来干预最终究输出结果了。
        action实例、xxxScope对象都在ValueStack内,直接使用${}调用。
        使用append标签时,没有所谓的将map和list集合对象连接起来的说法,创建map对象的语法为#{:,},如果省略冒号,则默认value为null。
        StackContext 中的对象可以理解为struts2标签库使用的专用对象,比如通过value属性调用:value="#xxx.xxx.xxx..."
        ValueStack是StackContext的顶级对象,可以不需要#前缀使用,StackContext中的某些已建立对象可以通过比如在某些标签中设置var属性调入根对象ValueStack,达到省略#号的目的。在实际中则可以理解为在标签内临时新建的一个或多个对象,通过var属性指定的名称访问value="xxx.xxx.xxx...."
        ${requestScope.xxx.name} 与 <s:propertyvalue="#request.xxx.name"/>(escapse属性设置为false) 表达的内容相同,都是输出request范围内的xxx对象的name属性值,且无论是否该值包含html代码;

Struts2中的ValueStack类相关推荐

  1. Struts2中的action类

    Struts2中的action类 action类在Struts2中承担了Model(模型)的角色,主要用于处理业务逻辑并存放HTTP请求处理过程中各个变量的值. 在Struts2里面,Action充当 ...

  2. 在Struts2中使用ValueStack、ActionContext、ServletContext、request、session等

    目录(?) [-] ValueStack 如何得到值栈: 如何将对象存入值栈: 让值栈执行表达式来获得值: 在JSP中跳过栈顶元素直接访问第二层: 在JSP中访问值栈对象本身(而不是它们的属性) Ac ...

  3. Struts2中XXX.Action类中方法的访问(三种方式)

    第一种:使用action标签的method属性(常规方法) Action类: package cn.tedu.web;import org.springframework.context.annota ...

  4. Struts2中的Action类(解耦方式,耦合方式)

    一.解耦方式 特点:对web资源进行了封装,便于单元测试. 实现:ActionContext和接口方式 1.ActionContext 特点:Action执行的上下文对象.保存了执行Action所需要 ...

  5. WEB Struts2 中OGNL的用法

    2019独角兽企业重金招聘Python工程师标准>>> User对象属性获取 如User中有username和password字段 获取username属性<s:propert ...

  6. Strus2中关于ValueStack详解

    什么是ValueStack 它是一个接口com.opensymphony.xwork2.util.ValueStack.我们使用它是将其做为一个容器,用于携带action数据到页面.在页面上通过ogn ...

  7. Struts2中OGNL,valueStack,stackContext的学习

    http://jwx0925.iteye.com/blog/630335 [/color]学习Struts2,一直不明白表单中的值是怎么传给Action的,上网查了些资料,基本了解了!下面基本是从几个 ...

  8. Struts2中jsp前台传值到action后台的三种方式以及valueStack的使用

    struts2中的Action接收表单传递过来的参数有3种方法: 如,登陆表单login.jsp: 1 <form action="login" method="p ...

  9. 深入分析JavaWeb Item48 -- Struts2中OGNL表达式与ValueStack

    一.OGNL表达式 1.OGNL表达式介绍 OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目. Struts2框架使用OGNL作 ...

  10. Struts2中ValueStack结构和总结

    [ValueStack和ActionContext的关系] 首先,从结构上来看ValueStack是ActionContext的一个组成部分,是对ActionContext功能的扩展.ActionCo ...

最新文章

  1. LoadRunner监控Linux
  2. linux c 延迟函数 sleep usleep 使用区别
  3. 笔记本电脑性价比排行2019_办公笔记本电脑排名2019 五款适合办公的笔记本电脑推荐...
  4. python中self_一篇文章让你彻底搞清楚Python中self的含义
  5. PMcaff课堂:10年经验的产品大咖眼中的社交产品是这样的
  6. FFmpeg 硬件加速方案概览 (上)
  7. 国科大高级人工智能9-模糊数学和遗传算法
  8. websocketpp 最简单的客户端 一
  9. jQuery easyui中combox 自定义样式 去掉下拉框的空白
  10. vb还是python强大-最难学的七大编程语言,VB 第一,Python垫底,看你学的排第几...
  11. SourceTree Win10 安装不成功解决过程记录
  12. Cannot read property 'scrollWidth' of null
  13. niceScroll相关配置参数
  14. 三菱无机房电梯故障代码查询_三菱无机房电梯的故障代码
  15. 生如夏花之绚烂,Lisp风格的Lialang诞生了!
  16. 阿里巴巴矢量图引入步骤
  17. 微信公众号token验证问题
  18. conda关于频道和存储包pkgs的详解(副pytorch环境安装)
  19. 服务器局域网无法访问共享文件夹,科学网—局域网共享文件夹不能访问 - 陈芳林的博文...
  20. 斯人已去,因荣永存(下)

热门文章

  1. 《Windows via C/C++》学习笔记 —— 内核对象的“线程同步”之“事件内核对象”...
  2. 基于STM32F103ZET6 HC_SR501人体红外感应
  3. .NetCore Session.Redis (转载)
  4. Sublime Text 插件 【转】
  5. ubuntu openStack icehouse dashboard theme自定义
  6. Model to Text工具Acceleo使用教程(六)——模板服务
  7. 注册、注销与结束session变量
  8. 19.看板方法---变异性的根源
  9. 21.卷1(套接字联网API)---多播
  10. 8.UNIX 环境高级编程--进程控制