深入Struts2

it2022-05-06  0

值栈 值栈是对应每一个请求的轻量级的内存数据中心,其实也是上一章讲的数据流元素ActionContext与ValueStack. 从广义上讲,值栈就是ActionContext,它是action的上下文环境.而从狭义上讲,值栈仅仅是指ValueStack. ValueStack是ActionContext的一个组成部分. 数据流有两个特性:数据和流.数据强调的是作为一个载体,流强调数据的访问和传输. ActionContext作为数据的载体,既负责数据的存储也负责数据共享. 而ValueStack是一个具备表达式引擎计算能力的数据结构,是为了解决数据访问和数据传输而定义得到特殊形象. 所以,Xwork将ValueStack置于ActionContext中的目的在于为静态的数据添加动态计算的功能.

ActionContext ActionContext是action的上下文环境.ActionContext真正的数据存储空间,是Map类型的变量context.ActionContext将所有的数据对象都以特定的键值存储与context之中.同时为了方便,提供了一些取值的快捷方式,如getValueStack,getSession.

2.1 数据共享 在数据共享的时候,如何保证"线程安全"呢.

public class ActionContext implements Serializable { static ThreadLocal actionContext = new ThreadLocal(); …… } 从源码可以看出,在ActionContext内部封装了一个ThreadLocal实例,而ThreadLocal实例所操作和存储的对象,又是ActionContext.这保证了其线程安全.

2.2 数据存储 ActionContext中存放了很多内容(包括action自身),大致可以分为两类: 对XWork框架对象的访:getContainer,getValueStack,getActionInvocation等等… 对数据对象的访问:getApplication,getSession,getParameters,getName等等… 值得注意的是,ActionContext对数据对象的访问,得到的都是一个Map对象而不是类似HttpSession或者ServletContext这样纯正的Web容器对象.这主要还是因为Xwork与Web容器的解耦. 解耦之后可以对两个方面做到更好:

被封装后的SessionMap等对象,进一步保证数据访问的线程安全性. 保持所有存储对象的Map结构,都有统一的数据访问方式. 当然我们还有更多存储数据的方式. 使用xxxAware接口:可以使用类似SessionAware,RequestAware之类的接口通过使用IoC/DI来为Action注入Map. 使用ServletActionContext:这个类直接继承了ActionContext,并且它能直接取到Servlet的相关对象,例如getRequest取到的就是HttpServletRequest.同时,使用这个子类同样可以通过IOC/DI的方式注入,不过好像显得多此一举.

ValueStack 3.1 OGNL OGNL是对象图导航语言Object-Graph Navigation Language的缩写,它是一种功能强大的表达式语言(Expression Language,简称为EL),通过它简单一致的表达式语法,可以存取对象的属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能.它使用相同的表达式去存取对象的属性. 所谓对象图,即以任意一个对象为根,通过OGNL可以访问与这个对象关联的其它对象.

Emp emp=new Emp(); DepartMent department=new DepartMent(); Enterprise enterprise=new Enterprise(); enterprise.setName(“A”); department.setEnterPrise(enterprise); emp.setDepartment(department); 那么利用OGNL导航就可以是

String value=(String)Ognl.getValue(“department.enterprise.name”,emp); 第一个参数是OGNL表达式,而第二个参数则是root对象. 由于OGNL中不支持多个root对象,所以如果要访问多个不相干的对象,就需要一个context上下文对象,它是一个Map类型的对象.

Emp emp=new Emp(); emp.setName(“张三”); Emp emp2=new Emp(); emp2.setName(“李四”); Emp emp3=new Emp(); emp3.setName(“王五”); Map context=new HashMap(); context.put(“e1”,emp); context.put(“e2”,emp2); String value=(String)Ognl.getValue("#e1.name+’,’+#e2.name+’,’+name",context,emp3); 这里把emp1和emp2存储到map中,取值时候需要加#,而放在root中的emp3则可以直接取值. OGNL还可以访问数组与集合,如果数组与集合在context中,那么类似如下取值

Ognl.getValue("#list[0]",context,root); Ognl.getValue("#array[0]",context,root); Ognl.getValue("#map[‘key’]",context,root); Ognl.getValue("#map.key",context,root); 如果是存放在root中的,那么就类似#root[0].value 以上方式可以组合使用.放到Struts2中考虑一些复杂的例子:

要获取Session中一个key值为“users”的List,对应的OGNL应为#session[‘users’],或者#session.users 要操作这个List的第3个元素,对应的OGNL应为#session[‘users’][2],或者#session.users[2] 要操作这个对象的userId属性,对应的OGNL应为#session[‘users’][2].userId,或者#session.users[2].userId 另外OGNL还可以进行赋值操作,直接获取root对象的方法,或者用@符号获取静态变量和方法等等,更多的知识可以去

3.2 ValueStack valuestack是对OGNL的一个扩展.我们知道OGNL有三要素,表达式,context对象以及root对象.而valuestack的扩展是针对root对象的.主要是,valuestack可以将一组对象都视为root对象,而在原生ognl中,root对象只有一个. valuestack从抽象层面上讲是一个栈,后入先出的链表结构.而valuestack实际上是一个接口,OgnlValueStack是其实现类. 观察源码可知,OgnlValueStack起核心作用的是一个叫CompoudRoot的数据结构,而它继承于ArrayList. 知道了ValueStack的数据结构后,来看看其对OGNL计算规则的影响. 由于可以有多个root对象(包括action本身),在进行表达式匹配的时候,从栈的顶端开始自上而下对每个栈内元素进行遍历匹配计算 .返回第一个成功匹配的结果. 另外,有两个重要的概念:栈顶元素和子栈. 所谓栈顶元素,就是可以通过[0]进行访问的元素,同时也可以通过top进行访问. 而子栈,就是出去栈顶元素以外的栈结构.[n]表示除去栈结构中前n个元素之后所构成的栈. 一个大小为N的ValueStack,除了自身,有N-1个子栈 每一个子栈自身也是一个ValueStack,构成递归的数据结构 显然我们可以用top访问第一个元素,用[1].top访问第二个元素.

水乳交融的ActionContext与ValueStack 水乳交融用来描述两者之间的关系.在学习的时候,这种密切会带来一些困扰,至少我之前是的. ActionContext的创建,总是伴随着ValueStack的创建 紧接着ValueStack的创建就是ActionContext的创建,而ActionContext的创建以ValueStack的上下文环境作为参数.两者几乎是相同时刻创建出来的. ValueStack的上下文环境与ActionContext的数据存储空间一致 深圳网站建设www.sz886.com

最新回复(0)