从调试角度理解ActionContext、OgnlContext、OgnlValueStack的关系
被调试代码:
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的关系的更多相关文章
- 从tcp原理角度理解Broken pipe和Connection reset by peer的区别
从tcp原理角度理解Broken pipe和Connection reset by peer的区别 http://lovestblog.cn/blog/2014/05/20/tcp-broken-pi ...
- 转:如何学习SQL(第二部分:从关系角度理解SQL)
转自:http://blog.163.com/mig3719@126/blog/static/285720652010950825538/ 6. 从关系角度理解SQL 6.1. 关系和表 众所周知,我 ...
- 从npm 角度理解 mvn 的 pom.xml
从npm 角度理解 mvn 的 pom.xml pom -- project object model. 用于描述项目的配置: 基础说明 依赖 如何构建运行 类似 node.js 的 package. ...
- 从极大似然估计的角度理解深度学习中loss函数
从极大似然估计的角度理解深度学习中loss函数 为了理解这一概念,首先回顾下最大似然估计的概念: 最大似然估计常用于利用已知的样本结果,反推最有可能导致这一结果产生的参数值,往往模型结果已经确定,用于 ...
- 以环形角度理解php数组索引
array_slice ( array $array , int $offset [, int $length = NULL [, bool $preserve_keys = false ]] ) : ...
- 从相亲的角度理解 K8S 的 Node Affinity, Taints 与 Tolerations
这是昨天晚上阅读园子里的2篇 k8s 博文时产生的想法,在随笔中记录一下. 这2篇博文是 K8S调度之节点亲和性 与 K8S调度之Taints and Tolerations . 如果我们把 node ...
- 从源码角度理解Java设计模式——装饰者模式
一.饰器者模式介绍 装饰者模式定义:在不改变原有对象的基础上附加功能,相比生成子类更灵活. 适用场景:动态的给一个对象添加或者撤销功能. 优点:可以不改变原有对象的情况下动态扩展功能,可以使扩展的多个 ...
- IL角度理解C#中字段,属性与方法的区别
IL角度理解C#中字段,属性与方法的区别 1.字段,属性与方法的区别 字段的本质是变量,直接在类或者结构体中声明.类或者结构体中会有实例字段,静态字段等(静态字段可实现内存共享功能,比如数学上的pi就 ...
- IL角度理解for 与foreach的区别——迭代器模式
IL角度理解for 与foreach的区别--迭代器模式 目录 IL角度理解for 与foreach的区别--迭代器模式 1 最常用的设计模式 1.1 背景 1.2 摘要 2 遍历元素 3 删除元素 ...
随机推荐
- PIR人体检查
1.无变化时电压为0.8V左右 2.当检查到人体活动,电压保持为-1.3V,直到运动停止. 下面是示波器的截图
- iOS: 如何获取ios设备的当前IP地址
有的时候,我们项目上线后,需要根据ip地址去统计不同地区的用户情况,此时IP地址的收取显得尤其重要,一般情况下,在用户登录时去获取用户的ip是准确的,当然实时追踪ip的变化而统计是更安全可靠的. ip ...
- DICOM中的入门概念
DICOM标准是医学影像界技术人员逃不掉的标准.本系列专题是JATI对DICOM标准的阐述,力图使PACS管理员和软件工程师都能理解. DICOM标准的提出者DICOM标准委员会是ISO组织的合作者. ...
- 用mysqlslap对MySQL进行压力测试
MySQL5.1地的确提供了好多有力的工具来帮助我们DBA进行数据库管理.现在看一下这个压力测试工具mysqlslap.关于他的选项手册上以及--help介绍的很详细.我解释一下一些常用的选项.这里要 ...
- PHP防抓取数据curl 解决方法
1.使用Snoopy或curl传搜索引擎爬虫的USERAGENT值. 查看搜索引擎爬虫的USERAGENT值:http://www.cnblogs.com/grimm/p/5068092.html ( ...
- Linux下性能监控的三把军刀
Linux主机怎么管,十八般兵器件件都可以算得上是瑞士军刀,称手的兵器一两件即可,最常用的,莫过于stat家族三兄弟吧. 计算机主要资源是什么?CPU.内存和磁盘?尽管现在云计算技术有多普及,查看一个 ...
- Mac版小黑屋提示无法确认开发者身份的解决办法
Mac版小黑屋提示无法确认开发者身份的解决办法 学习了:https://jingyan.baidu.com/article/37bce2be703fa21003f3a259.html 需要按住cont ...
- Python——管理属性(1)
管理属性 这里将展开介绍前面提到的[属性拦截]技术.包含下面内容: [1]__getattr__和__setattr__方法.把没有定义的属性获取和全部的属性赋值指向通用的处理器方法 [2]__get ...
- wap、app移动端页面常用html标签汇总
1.section 将内容组织到精确的语义块,表示页面的一部分. 2.article article表示网页的一个文章.故事. 3.header (1)用在整页的页头 (2)section或者arti ...
- java线程同步: synchronized详解(转)
Java语言的关键字,当它用来修饰一个方法或者一个代码块的时候,能够保证在同一时刻最多只有一个线程执行该段代码. 一.当两个并发线程访问同一个对象object中的这个synchronized(this ...