1.什么是值栈?
//获取值栈的三种方式
// ValueStack vs = (ValueStack) ServletActionContext.getRequest().getAttribute("struts.valueStack");
//ValueStack vs2 = (ValueStack) ServletActionContext.getRequest().getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
ValueStack vs= ServletActionContext.getContext().getValueStack();
2.值栈的内部结构是什么?
3.值栈对象的创建。ValueStack 和ActionContext的关系
A> ActionContext:(里面必须传进来一个map集合,传进来的map赋值给成员变量)
1>里面有一个属性: private Map<String, Object> context;
2>这是他的构造方法: public ActionContext(Map<String, Object> context) {
this.context = context;}
ValueStack:他本是就是一个大的map
//创建值栈
ValueStack stack = dispatcher.getContainer().getInstance(ValueStackFactory.class).createValueStack();
//往值栈里面压值,这些值都是一些map集合
stack.getContext().putAll(dispatcher.createContextMap(request, response, null, servletContext));
//ctx就是ActionContext这个对象。这是他的构造方法,创建他的时候需要一个map集合。里面传的就是值栈这个大Map
ctx = new ActionContext(stack.getContext());
根据上述:ActionContext中存在一个Map集合,该Map集合和ValueStack的contextMap是同一个地址。
B> * ActionContext是绑定到当前的线程上,那么在每个拦截器或者Action中获取到的ActionContext是同一个。
static ThreadLocal<ActionContext> actionContext = new ThreadLocal<ActionContext>();
ThreadLocal:底层其实是一个map集合。key:是当前线程 value: actionContext(目的就是保证获取到的actionContext是同一个)
C>ActionContext中可以获取到ValueStack的引用。开发的时候经常用的方法
ValueStack stack = ServletActionContext.getContext().getValueStack();
D>值栈对象什么时候被创建的: * 值栈对象是请求时创建的
4.如何获得值栈的对象
OgnlValueStack的push():
public void push(Object o) {
root.push(o);--->root其实就是CompoundRoot(其实就是ArrayList集合)
}
CompoundRoo的push方法,其实就是集合添加元素
public void push(Object o) {
add(0, o);
}
往值栈里面存值:
push和set的异同点:
相同点:都操作的是栈顶的元素
不同点:
push(Object o):直接压栈,不管是什么。 push方法的底层调用root对象的push方法(把元素添加到0位置)
set(String key, Object obj) :源码获取map集合(map有可能是已经存在的,有可能是新创建的),
把map集合push到栈顶,再把数据存入到map集合中。
5.向值栈保存数据
两种方式
> valueStack.push(Object obj);
* push方法的底层调用root对象的push方法(把元素添加到0位置)
> valueStack.set(String key, Object obj);
* 源码获取map集合(map有可能是已经存在的,有可能是新创建的),把map集合push到栈顶,再把数据存入到map集合中。
> 在jsp中 通过 <s:debug /> 查看值栈的内容
6.在jsp中获取值栈的数据
总结几个小问题:
> 访问root中数据 不需要#
> 访问context其它对象数据 加 #
> 如果向root中存入对象的话,优先使用push方法。
> 如果向root中存入集合的话,优先要使用set方法。
* 在OgnlContext中获取数据
> 在Action中向域对象中存入值
> request:<s:property value="#request.username"/>
> session:<s:property value="#session.username"/>
> application:<s:property value="#application.username"/>
> attr:<s:property value="#attr.username"/>
//获取值栈的三种方式
// ValueStack vs = (ValueStack) ServletActionContext.getRequest().getAttribute("struts.valueStack");
//ValueStack vs2 = (ValueStack) ServletActionContext.getRequest().getAttribute(ServletActionContext.STRUTS_VALUESTACK_KEY);
ValueStack vs= ServletActionContext.getContext().getValueStack();
2.值栈的内部结构是什么?
3.值栈对象的创建。ValueStack 和ActionContext的关系
A> ActionContext:(里面必须传进来一个map集合,传进来的map赋值给成员变量)
1>里面有一个属性: private Map<String, Object> context;
2>这是他的构造方法: public ActionContext(Map<String, Object> context) {
this.context = context;}
ValueStack:他本是就是一个大的map
//创建值栈
ValueStack stack = dispatcher.getContainer().getInstance(ValueStackFactory.class).createValueStack();
//往值栈里面压值,这些值都是一些map集合
stack.getContext().putAll(dispatcher.createContextMap(request, response, null, servletContext));
//ctx就是ActionContext这个对象。这是他的构造方法,创建他的时候需要一个map集合。里面传的就是值栈这个大Map
ctx = new ActionContext(stack.getContext());
根据上述:ActionContext中存在一个Map集合,该Map集合和ValueStack的contextMap是同一个地址。
B> * ActionContext是绑定到当前的线程上,那么在每个拦截器或者Action中获取到的ActionContext是同一个。
static ThreadLocal<ActionContext> actionContext = new ThreadLocal<ActionContext>();
ThreadLocal:底层其实是一个map集合。key:是当前线程 value: actionContext(目的就是保证获取到的actionContext是同一个)
C>ActionContext中可以获取到ValueStack的引用。开发的时候经常用的方法
ValueStack stack = ServletActionContext.getContext().getValueStack();
D>值栈对象什么时候被创建的: * 值栈对象是请求时创建的
4.如何获得值栈的对象
OgnlValueStack的push():
public void push(Object o) {
root.push(o);--->root其实就是CompoundRoot(其实就是ArrayList集合)
}
CompoundRoo的push方法,其实就是集合添加元素
public void push(Object o) {
add(0, o);
}
往值栈里面存值:
push和set的异同点:
相同点:都操作的是栈顶的元素
不同点:
push(Object o):直接压栈,不管是什么。 push方法的底层调用root对象的push方法(把元素添加到0位置)
set(String key, Object obj) :源码获取map集合(map有可能是已经存在的,有可能是新创建的),
把map集合push到栈顶,再把数据存入到map集合中。
5.向值栈保存数据
两种方式
> valueStack.push(Object obj);
* push方法的底层调用root对象的push方法(把元素添加到0位置)
> valueStack.set(String key, Object obj);
* 源码获取map集合(map有可能是已经存在的,有可能是新创建的),把map集合push到栈顶,再把数据存入到map集合中。
> 在jsp中 通过 <s:debug /> 查看值栈的内容
6.在jsp中获取值栈的数据
总结几个小问题:
> 访问root中数据 不需要#
> 访问context其它对象数据 加 #
> 如果向root中存入对象的话,优先使用push方法。
> 如果向root中存入集合的话,优先要使用set方法。
* 在OgnlContext中获取数据
> 在Action中向域对象中存入值
> request:<s:property value="#request.username"/>
> session:<s:property value="#session.username"/>
> application:<s:property value="#application.username"/>
> attr:<s:property value="#attr.username"/>
> parameters:<s:property value="#parameters.cid"/>
=========================================
为什么EL也能访问值栈中的数据?
* StrutsPreparedAndExecuteFilter的doFilter代码中 request = prepare.wrapRequest(request);
> 对Request对象进行了包装 ,StrutsRequestWrapper
> 增强了request的 getAttribute
Object attribute = super.getAttribute(s);
if (attribute == null) {
attribute = stack.findValue(s);
}
> 访问request范围的数据时,如果数据找不到,去值栈中找
> request对象 具备访问值栈数据的能力 (查找root的数据)