OgnlContext、OgnlValueStack、CompoundRoot的数据结构

在上一篇从struts2的action中看ActionContext的存储结构中已经看到ActionContext.getActionContext.getContextMap()得到的map的大体结构类型。

这篇将从源代码层面进行分析。


一、回顾

在上一篇中看到ActionContext中存放的是OgnlContext这个类,这个类的源代码,在OgnlContext源码分析中已经将的比较清楚了。那么值栈ValueStack到底是什么鬼。

二、ValueStack

我们就从ActionContext类中的getValueStack()方法入手:

    public ValueStack getValueStack() {
        return (ValueStack) get(VALUE_STACK);
    }
    public Object get(String key) {
        return context.get(key);
    }
看来要得到这个值栈,还是要从context这个所谓的map中去取。

这个context的实际类型是OgnlContext,还记得OgnlContext是由一系列保留的Object以及一个HashMap共同维护的:先找保留的key,找不到再去HashMap中去找。

这里的key为VALUE_STACK,是要从HashMap中取的(还记得这个HashMap的属性名为_values)

也就是说,ValueStack是藏在OgnlContext中_values属性的hashMap的key为VALUE_STACK对应的值中。


在action中打印一下ValueStack真实的类:

System.out.println(ActionContext.getContext().getValueStack().getClass());

//结果
//class com.opensymphony.xwork2.ognl.OgnlValueStack
那么我们就打开OgnlValueStack来看里面的源代码。


三、OgnlValueStack

1.两个重要的属性

    CompoundRoot root;
    transient Map<String, Object> context;

2.先来说说这个叫做root的属性:

CompoundRoot比较陌生,打开类看一下:

/*
 * Copyright 2002-2006,2009 The Apache Software Foundation.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.opensymphony.xwork2.util;

import java.util.ArrayList;
import java.util.List;


/
 * A Stack that is implemented using a List.
 * 
 * @author plightbo
 * @version $Revision: 894090 $
 */
public class CompoundRoot extends ArrayList {

    public CompoundRoot() {
    }

    public CompoundRoot(List list) {
        super(list);
    }


    public CompoundRoot cutStack(int index) {
        return new CompoundRoot(subList(index, size()));
    }

    public Object peek() {
        return get(0);
    }

    public Object pop() {
        return remove(0);
    }

    public void push(Object o) {
        add(0, o);
    }
}

这样看来,不要太简单,就把他当做是一个Stack来看把。

3.再来看看这个context属性:

在OgnlValueStack的构造方法都调用的

    protected void setRoot(XWorkConverter xworkConverter, CompoundRootAccessor accessor, CompoundRoot compoundRoot,
                           boolean allowStaticMethodAccess) {
        this.root = compoundRoot;
        this.securityMemberAccess = new SecurityMemberAccess(allowStaticMethodAccess);
        this.context = Ognl.createDefaultContext(this.root, accessor, new OgnlTypeConverterWrapper(xworkConverter), securityMemberAccess);
        context.put(VALUE_STACK, this);
        Ognl.setClassResolver(context, accessor);
        ((OgnlContext) context).setTraceEvaluations(false);
        ((OgnlContext) context).setKeepLastEvaluation(false);
    }
这里又多出一个Ognl工具类,先不用管具体的代码,只要知道Ognl.createDefaultContext他是产生了一个OgnlContext对象,然后返回。

应该还记得,OgnlContext这个类实现了Map接口,Map context=ognlContext是完全没问题。

到这里也要记住,root他放的是一个stack,context放的是一个OgnlContext

四、OgnlContext

再回来看下Ongl这个工具类是怎么初始化一个OgnlContext对象的。

    public static Map createDefaultContext( Object root, ClassResolver classResolver, TypeConverter converter, MemberAccess memberAccess )
    {
        return addDefaultContext( root, classResolver, converter, memberAccess, new OgnlContext() );
    }

public static Map addDefaultContext( Object root, ClassResolver classResolver, TypeConverter converter, MemberAccess memberAccess, Map context )
    {
        OgnlContext     result;

        if (!(context instanceof OgnlContext)) {
            result = new OgnlContext();
            result.setValues(context);
        } else {
            result = (OgnlContext)context;
        }
        if (classResolver != null) {
            result.setClassResolver(classResolver);
        }
        if (converter != null) {
            result.setTypeConverter(converter);
        }
        if (memberAccess != null) {
            result.setMemberAccess(memberAccess);
        }
        result.setRoot(root);
        return result;
    }

参数太多不管,只要看到root传过来就行了。

然后在return之前调用了setRoot方法,什么意思呢?

在OgnlContext中的_root属性上放的是一个Stack,就是叫做CompoundRoot的那个东西。


五、总结

可以看到,OgnlContext、OgnlValueStack之间的互相引用有些复杂:

OgnlContext=保留字+HashMap

HashMap=key:value...+value_stack:ognlValueStack

保留字=其他保留字+CompoundRoot


OgnlValueStack=CompoundRoot+OgnlContext

上丑图:

OgnlContext、OgnlValueStack、CompoundRoot的数据结构

更多相关文章
  • 来源:中国台湾网中国台湾网8月5日讯 台湾宏达电子公司(亦称HTC)一名男员工,日前趁女性友人到桃园公司会客时,两人在会议室嘿咻,亲热过程遭人用手机偷拍.视频长达一分钟,与日前大陆出现的"优衣库试衣间不雅视频"极为相似.宏达电子方面目前已证实此事,并表示已完成内部调查,解雇该员工 ...
  • Android的对话框有两种:PopupWindow和AlertDialog.它们的不同点在于: AlertDialog的位置固定,而PopupWindow的位置可以随意 AlertDialog是非阻塞线程的,而PopupWindow是阻塞线程的 PopupWindow的位置按照有无偏移分,可以分为 ...
  • 一.生成APK Build-->Generate Signed APK... 以上是创建密钥库及密钥,创建后会自动选择刚创建的密钥库和密钥.若已经创建过输入password即可. 若第一次或需要新建点击Create new... Key store path:密钥库文件的地址 Password ...
  • 构思: 在首页(MainPage)上定义一个代表应用版本号的字符串,网站上写一个接口(让这个字符串与你的最新应用版本号进行比较,返回应用的ContentIdentifier与操作状态什么的). 一.wp客户端 using System; using System.Collections.Generi ...
  • 几年前,我们这样写前端代码: <div id="el" style="......" onclick="......">测试</div> 慢慢的,我们发现这样做的很多弊端,单就样式一块,改一个样式会涉及到多处调整,所 ...
  • 如何安装WIN7系统过程图解   下面请看WIN7安装的全过程:   首先在开启电脑时,要进入主板BIOS设置成光盘启动:     重新启动电脑,出现提示Enter Boot from CD rom,按Enter就进入安装界面:  www.2cto.com         根据中文提示进行操作就是了 ...
一周排行
  • 结构型设计模式概述  结构型设计模式用于处理类或对象之间的组合,即描述类和对象之间怎样组织起来形成大的结构,从而实现新的功能. 实现的机制: 结构型对象模式采用组合/聚合机制来组合类,包括桥梁模式(Bridge).组 ...
  • 来源ispublic.com Google上貌似找不到利用开源软件实现上网行为审计的文章--这也难怪,开源在国内并不流行,而上网行为审计在国外也不流行.不过这并不代表不能实现,IS in Public通过本文提供一种 ...
  • "在腾讯和阿里布局贵州的步伐完成后,作为区域中心城市的贵阳,将在大数据应用方面领先于全国",冯仁超说,"有了政府成功的示范效应,贵州省在大数据产业上将会有更多的成长空间".本报 ...
  • 新公司,维护一个旧代码:   一个内存越界问题: c = *(sText+nLeng+1); // 读随便你 *(sText+nLeng+1) = 0; // 越界两个字符,写坏内存 这会导致, 在无压力正常情况下运 ...
  • 1.意图 将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化:对请求排队或记录请求日志,以及支持可撤销的操作. 2.别名 动作(Action),事务(Transaction) 3.动机 有时必须向某对象 ...
  • 转载请注明出处:http://blog.csdn.net/guoyjoe/article/details/12361287正确答案:AD 来自官方对CHANGE FAILURE command的语法解释,如下图所示: ...
  • 功能开发与优化: 1. java客户端下载功能改善 2. 提高下载速度,不同的手机 速度提升不一样.一般有2K以上提升   3. 增加快捷键打开预读链接功能    4. 下载管理界面优化   5. 网址输入适配加入对 ...
  • 8月4日,金山手机安全中心截获Android手机木马程序Netsnd的多个变种.Netsnd手机木马窃取的信息包括:手机串号.手机内置软件列表.手机型号,并可伺机推广手机流氓软件.金山手机安全专家建议水货Androi ...
  • Camera和手机之间通过中转服务器来转发数据.如果任何一方非正常退出时, 会导致select函数执行异常.  不需要在服务器定时检查每个socket描述符.  解决方法如下: 1)   手机播放视频时每个30秒发送 ...
  • 1.UTF-8编码实验 严 二进表示 100111000100101 按UTF-8格式,6位分组拆分      100   111000   100101 UTF-8编码                        ...