被调试代码:

   package
web;

import
java.util.Map;

import
javax.servlet.http.HttpServletRequest;

import
org.apache.struts2.ServletActionContext;
    import
org.apache.struts2.StrutsConstants;
    import
org.apache.struts2.StrutsStatics;

import
com.opensymphony.xwork2.Action;
    import
com.opensymphony.xwork2.ActionContext;
    import
com.opensymphony.xwork2.ActionSupport;
    import
com.opensymphony.xwork2.util.ValueStack;

public class
LoginAction implements Action {

private
String userName;

public void
setUserName(String userName) {
   
   
   
this.userName = userName;
   
    }

public
String getUserName() {
   
   
    return
userName;
   
    }

public
String doLogin() {
   
   
   
ActionContext cxt = ActionContext.getContext();
   
   
    ValueStack
vstack = cxt.getValueStack();
   
   
   
   
   
   
//返回前下断点
   
   
    return
Action.SUCCESS;
   
    }

@Override
   
    public
String execute() throws Exception {
   
   
    // TODO
Auto-generated method stub
   
   
    return
null;
   
    }
   
   
    }

访问url地址:http://localhost:8080/strutsdemo/doLogin.action?userName=jiang

被观察的表达式:
 ActionContext.getContext();
 ActionContext.getContext().getValueStack();
观察结论:
  1、从ActionContext对象中可以取得OgnlValueStack对象

2、ActionContext中持有的属性context和OgnlValueStack对象中持有的对象属性context的类型都是OgnlContext类型,其实是同一个对象

3、Ognl持有一个root属性,这个属性存放的是Struts
Ognl的根对象
   OGNL三要素:

  1.expression 求值表达式——首先会被解析成对象树
  2.root object 根对象——默认的操作对象
  3.context OGNL执行环境——OGNL执行的上下文环境
OGNL context是一个Map结构,ognl.OgnlContext类implements
Map接口,root对象也在context里面,并且做这一个特殊的对象处理,具体表现为对root 
对象的操作不需要加#指示符号(并且加上了#一定取不到root对象里面的值)。
继续展开root属性节点:
4、从root的对象名称我们都可以看出他表达的含义:CompoundRoot表示符合的根对象,根据原始的Ognl,根对象是单个对象。而
Struts2扩展了这个概念,这个跟对象不在仅仅只代表单个对象而可以是多个对象,这里我们从调试中可以看出它包含当前Action对象和
DefaultTextProvider对象。CommpoundRoot为何可以包含多个对象?

很简单,因为它集成自ArrayList集合,从下面扩展的方法中可以看出它实现了对集合栈数据结构方式的存取能力。我们看看另一段关于这个CompoundRoot的解释:

There can be many "root" objects.
XWork中的表示根对象是CompoundRoot对象。CompoundRoot类extends
ArrayList类。因为是一个List,里面可以放置多个对象,而这些对象经过XWork的改进对于OGNL表达式引擎来说都是root
objects。XWork has a special OGNL PropertyAccessor that will
automatically look at the all entries in the stack (in
fact the CompoundRoot list) (from the top down) until it finds an object with the property you are
looking for.

再来重点观察OgnlContext这个非常重要的容器:


5、OgnlContext实现了Map接口,这里_values私有属性存放的是Map的键值对信息,我们展开里面的table节点观察里面存放了什么秘密:

6、我只是展开了部分,从中我们可以看出Struts2放置请求表单参数、请求对象本身、session对象、application
对象和application对象属性等等到OGNL栈(事实上放在OgnlContex).Struts 2 places request
parameters and request, session, and application attributes on the
OGNL stack (in fact the OGNL context).

小结:
 1、Struts2数据传输DataTransfer的核心对象是OgnlValueStack、
OgnlContext,OgnlValueStack持有OgnlContext和root对象,注意:OgnlContext和root对象是
ognl的核心要素【参考我的转载http://blog.csdn.net/jiangtongcn/article/details/7669871】。
 2、OgnlContext这个容器存放了所有本次Web请求响应的所有相关信息对象,root的存放的多个根对象。

3、Struts2
Ognl表达式的解析主要依赖OgnlValueStack去完成。表达式中如果没有带#,如:“emp.name”,那么OgnlValueStack
将去根对象中去遍历每个对象看是否某个对象的方法是否匹配"emp.getName()",匹配则执行,否则跑出OgnlExcpetion异常。如果带有#,则去Ognl上下文中去寻找是否有满足emp.name属性的key,有责返回。

4、Struts2的ActionContext类只是为了访问ValueStack而提供的一个Facade【门面设计模式】,为程序员访问各种信息提供一个一致的界面。

5、使用Ognl表达式从OgnlValueStack中取出值:
 代码:

public class
LoginAction implements Action {

private
String userName;

public void
setUserName(String userName) {
   
   
   
this.userName = userName;
   
    }

public
String getUserName() {
   
   
    return
userName;
   
    }

public
String doLogin() {
   
   
   
ActionContext cxt = ActionContext.getContext();
   
   
    ValueStack
vstack = cxt.getValueStack();

Employee
emp1 = new Employee(){{
   
   
   
   
setUserName("小强");
   
   
    }};
   
   
    Employee
emp2 = new Employee(){{
   
   
   
   
setUserName("秋香");
   
   
    }};
   
   
   
   
   
   
//向栈栈顶压入两个员工对象
   
   
   
vstack.set("emp1", emp1);
   
   
   
vstack.setValue("emp2",emp2);
   
   
   
   
   
   
   
   
   
//向request请求参数集合中放入一个属性和属性值
   
   
   
cxt.getParameters().put("pwd","123123");
   
   
   
//向request请求中放入一个属性和属性值
   
   
   
((Map<String,Object>)cxt.get("request")).put("pwd",
"112233");
   
   
   
//向Session中放入一个属性和属性值
   
   
   
cxt.getSession().put("pwd", "112211");
   
   
   
//向Application中放入一个属性和属性值
   
   
   
cxt.getApplication().put("pwd", "112222");
   
   
   
   
   
   
//向OgnlContext上下文放入一个属性和属性值
   
   
   
cxt.put("pwd", "123321");
   
   
   
   
//   
   
   
   
   
   
   
   
//返回前下断点
   
   
    return
Action.SUCCESS;
   
    }

@Override
   
    public
String execute() throws Exception {
   
   
    // TODO
Auto-generated method stub
   
   
    return
null;
   
    }
   
   
    }

class
Employee {

private
String userName;

public
String getUserName() {
   
   
    return
userName;
   
    }

public void
setUserName(String userName) {
   
   
   
this.userName = userName;
   
    }
    }

观察Ognl表达式的值:
   
   
   
System.out.println("vstack.findValue(\"userName\")
==>
"+vstack.findValue("userName"));

System.out.println("vstack.findValue(\"emp1.userName\")
==> "+vstack.findValue("emp1.userName"));
   
   
   
System.out.println("vstack.findValue(\"emp2.userName\")
==> "+vstack.findValue("emp2.userName"));
   
   
   
System.out.println("vstack.findValue(\"#parameters.userName\")
==>
"+vstack.findValue("#parameters.userName"));
   
   
   
System.out.println("vstack.findValue(\"#request.pwd\")
==> "+vstack.findValue("#request.pwd"));
   
   
   
System.out.println("vstack.findValue(\"#session.pwd\")
==> "+vstack.findValue("#session.pwd"));
   
   
   
System.out.println("vstack.findValue(\"#application.pwd\")
==> "+vstack.findValue("#application.pwd"));
   
   
   
System.out.println("vstack.findValue(\"pwd\") ==>
"+vstack.findValue("pwd"));
   
   
   
System.out.println("vstack.findValue(\"#attr.pwd\")
==> "+vstack.findValue("#attr.pwd"));

结果如下:
   
   
   
vstack.findValue("userName") ==> jiang
   
vstack.findValue("emp1.userName") ==> 小强
   
vstack.findValue("emp2.userName") ==> 秋香
   
vstack.findValue("#parameters.userName") ==>
[Ljava.lang.String;@1e9b48b
   
vstack.findValue("#request.pwd") ==> 112233
   
vstack.findValue("#session.pwd") ==> 112211
   
vstack.findValue("#application.pwd") ==>
112222
   
vstack.findValue("pwd") ==> 123321
   
vstack.findValue("#attr.pwd") ==> 112233

http://www.linuxso.com/architecture/34752.html

从调试角度理解ActionContext、OgnlContext、OgnlValueStack的关系的更多相关文章

  1. 从tcp原理角度理解Broken pipe和Connection reset by peer的区别

    从tcp原理角度理解Broken pipe和Connection reset by peer的区别 http://lovestblog.cn/blog/2014/05/20/tcp-broken-pi ...

  2. 转:如何学习SQL(第二部分:从关系角度理解SQL)

    转自:http://blog.163.com/mig3719@126/blog/static/285720652010950825538/ 6. 从关系角度理解SQL 6.1. 关系和表 众所周知,我 ...

  3. 从npm 角度理解 mvn 的 pom.xml

    从npm 角度理解 mvn 的 pom.xml pom -- project object model. 用于描述项目的配置: 基础说明 依赖 如何构建运行 类似 node.js 的 package. ...

  4. 从极大似然估计的角度理解深度学习中loss函数

    从极大似然估计的角度理解深度学习中loss函数 为了理解这一概念,首先回顾下最大似然估计的概念: 最大似然估计常用于利用已知的样本结果,反推最有可能导致这一结果产生的参数值,往往模型结果已经确定,用于 ...

  5. 以环形角度理解php数组索引

    array_slice ( array $array , int $offset [, int $length = NULL [, bool $preserve_keys = false ]] ) : ...

  6. 从相亲的角度理解 K8S 的 Node Affinity, Taints 与 Tolerations

    这是昨天晚上阅读园子里的2篇 k8s 博文时产生的想法,在随笔中记录一下. 这2篇博文是 K8S调度之节点亲和性 与 K8S调度之Taints and Tolerations . 如果我们把 node ...

  7. 从源码角度理解Java设计模式——装饰者模式

    一.饰器者模式介绍 装饰者模式定义:在不改变原有对象的基础上附加功能,相比生成子类更灵活. 适用场景:动态的给一个对象添加或者撤销功能. 优点:可以不改变原有对象的情况下动态扩展功能,可以使扩展的多个 ...

  8. IL角度理解C#中字段,属性与方法的区别

    IL角度理解C#中字段,属性与方法的区别 1.字段,属性与方法的区别 字段的本质是变量,直接在类或者结构体中声明.类或者结构体中会有实例字段,静态字段等(静态字段可实现内存共享功能,比如数学上的pi就 ...

  9. IL角度理解for 与foreach的区别——迭代器模式

    IL角度理解for 与foreach的区别--迭代器模式 目录 IL角度理解for 与foreach的区别--迭代器模式 1 最常用的设计模式 1.1 背景 1.2 摘要 2 遍历元素 3 删除元素 ...

随机推荐

  1. 查找链表中倒数第k个结点

    题目:输入一个单向链表,输出该链表中倒数第k个结点.链表的倒数第0个结点为链表的尾指针.链表结点定义如下: struct ListNode { int m_nKey; ListNode* m_pNex ...

  2. poj 3264 Balanced Lineup 题解

    Balanced Lineup Time Limit: 5000MS   Memory Limit: 65536KB   64bit IO Format: %I64d & %I64u Subm ...

  3. Tensorflow之调试(Debug) && tf.py_func()

    Tensorflow之调试(Debug)及打印变量 tensorflow调试tfdbg 几种常用方法: 1.通过Session.run()获取变量的值 2.利用Tensorboard查看一些可视化统计 ...

  4. Android选择/拍照 剪裁 base64/16进制/byte上传图片+PHP接收图片

    转载请注明出处:http://blog.csdn.net/iwanghang/article/details/65633129认为博文实用,请点赞,请评论,请关注.谢谢! ~ 老规矩,先上GIF动态图 ...

  5. Android:Volley源代码解析

    简单实例 Volley是一个封装HttpUrlConnection和HttpClient的网络通信框架,集AsyncHttpClient和Universal-Image-Loader的长处于了一身.既 ...

  6. FFmpeg视频处理必备

    http://ffmpeg.org/官网 A complete, cross-platform solution to record, convert and stream audio and vid ...

  7. web前端开发文档

  8. UML和模式应用学习笔记-2(迭代和进化式开发)

    一:什么是迭代和进化式开发 1:迭代和进化式开发:通常会在还没有详细定义所有需求的情况下假设开发开始,同时使用反馈来明确和改进演化中的规格说明: 2:迭代方法与较高的成功率.生产率和低缺陷率具有关系: ...

  9. MySQL主从架构之Master-Slave主从同步

    MySQL复制 MySQL复制是指将主库上的DDL和DML操作通过二进制日志传到从库上,使主库和从库上的数据保持同步 复制原理: 主服务器将更新写入二进制日志文件,并维护文件的一个索引来跟踪日志循环. ...

  10. Mac安装Myeclipse2015开发环境

    1.下载Myeclipse2015 链接: http://pan.baidu.com/s/1jHe8mFk 密码: qgeb 下载下来后,在安装的时候需要自己设置下安装目标,不然在破解的时候不是太好找 ...