从调试角度理解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 删除元素 ...
随机推荐
- boa cgi程序cgi_header: unable to find LFLF
ftp必须用二进制模式上传才可以 sqlite3 arm-linux-gcc hello.c -o hello.cgi -I /cgi/include -L /cgi/lib -static -lsq ...
- windows服务器提权前请先执行systeminfo命令
pr.巴西烤肉 对应补丁 好多朋友见到Windows服务器就祭出pr.巴西烤肉一气搞,忙完免杀又忙找可写目录,最后发现服务器打上了对应的补丁.笔者在提权前都会执行systeminfo命令,查看对应补丁 ...
- RSA大会播报 – 2014最佳安全博客提名
今年美国RSA大会将在这个月的23-28号举行,每年大会上都会评出过去一年来业内最佳安全博客(Security Bloggers Network Social Security Awards 2014 ...
- C#文件系统管理【转】
目录 前言 Directory类和DirectoryInfo类 File类和FileInfo类 Path类 前言 管理文件系统主要是对计算机中文件和目录的管理,例如,读取文件信息.删除文件和读取目录信 ...
- 通过Spring配置文件中bean中的property赋值
基本数据类型赋值-通过spring配置文件中bean中的property 扩展-以此方式可以通过配置为连接数据的属性赋值 1.如果是基本数据类型,可以通过setter方法为对象中的属性设置初始值,应用 ...
- SQL IN
here are some additional clause in the SQL language that can be used to simplify queries by decrease ...
- Android实战简易教程-第二十八枪(基于Bmob实现头像图片设置和网络上传功能!)
上一篇我们介绍了怎样由uri转换成String ,本文就用到了上篇文章的方法.以下我们介绍一下怎样设置头像后将头像图片上传到云端的方法,本文基于Bmob提供的服务. 看一下代码:(布局文件和前两篇文章 ...
- JAVA:连接池技术说明以及MVC设计模式理解
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdTAxMjgzMDgwNw==/font/5a6L5L2T/fontsize/400/fill/I0JBQk ...
- sass 和 css 互转网址
sass to css:https://www.sassmeister.com/ css to sass:http://css2sass.herokuapp.com/
- css3和html5的学习
本文是此链接的源代码.http://www.imooc.com/learn/77 关于的html5的使用: transition----含义是:过渡的过程,能够对各种属性设置变化. 有5中过渡的形式: ...