• Struts2学习笔记(十) OGNL


    OGNL介绍

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

    OGNL三要素

    (1).expression 求值表达式——首先会被解析成对象树

    (2).rootobject  根对象——默认的操作对象

    (3).context OGNL执行环境——OGNL执行的上下文环境

    OGNL context是一个Map结构,ognl.OgnlContext类implements Map接口,root对象也在context里面,并且做这一个特殊的对象处理,具体表现为对root  对象的操作不需要加#指示符号(并且加上了#一定取不到root对象里面的值).

    OGNL中的Ognl类提供了一下额静态方法用于对comtext中的对象进行操作。

    public static Object getValue(String expression, Map context, Object root, Class resultType)
                throws OgnlException
        {
            return getValue(parseExpression(expression), context, root, resultType);
    }
    public static void setValue(String expression, Map context, Object root, Object value)
                throws OgnlException
        {
            setValue(parseExpression(expression), context, root, value);
    }

    我们使用OGNL来做一个实例:

    public class OgnlTest {
    
        public static void main(String[] args) throws Exception {
            
            User user1 = new User();
            user1.setUserName("hello");
            
            User user2 = new User();
            user2.setUserName("world");
            
            OgnlContext context = new OgnlContext();
            context.put("user1",user1);
            context.put("user2",user2);
            context.setRoot(user1);
            
            Object o = Ognl.parseExpression("userName");
            
            Object obj = Ognl.getValue(o,context.getRoot());
            System.out.println(obj);
        }
    }

    这段程序执行结果能够将user1的userName获取出来,但是并不能获取到user2的任何信息。这是因为我们使用的getValue版本中没有传递context参数值,那么OGNL就会新建一个OgnlContext对象,并把我们的root对象放进去。那么此时在这个新建的context中除了我们的root对象之外就没有其他对象了,因此只能访问root对象。由于OgnlContext类实现了Map接口,我们可以直接使用Map对象来作为参数,Ognl会自动将我们传递的Map对象转换为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;
    
    }

    Struts2OGNL

    OGNL的功能非常强大,Struts2在原生的OGNL上又做了一些扩展。比如在Struts2中使用valueStack来作为数据存储的载体,并且在Strtus2扩展的OGNL中,root对象可以不只是一个。在Strtus2中的Root使用的是CompoundRoot对象,而CompoundRoot继承了ArrayList,所以他可以存储一系列的对象,这些对象可以看作是OGNL中的root对象。当我们当问某个属性时,CompoundRootAccessor对象实例会负责在CompoundRoot对象中找到包含我们指定属性的对象一般情况下我们只会接触到OGNL的一小部分功能,所以我们就主要学习一下我们可能会用到的知识点,如果要更加深入的学习OGNL,那么可以去看一下OGNL的官方文档,上面有非常详细的介绍。

    valueStack

    对于每个动作调用,Struts2在执行相应的动作方法之前会先创建一个名为valueStack的对象。valueStack用来保存该动作对象和其他对象。在对动作进行处理的过程中,拦截器需要访问valueStack,视图也需要访问valueStack才能显示动作和其他信息。

    valueStack的内部包含两个逻辑部分,一个叫做Object Stack,另一个叫做ComtextMap。Struts2将动作和相关对象压入Object Stack,把各种各样的映射关系(Map类型的对象)压入Comtext Map。

    其中的Object Stack中的对象都相当于OGNL中的”root”对象,因此对他们可以直接访问。如果要访问Context Map中的对象,那么就得在OGNL表达式前面加上”#”符号。如果没有加”#”,那么Struts2默认会在Object Stack中进行搜索。

    Strut2会把下面的这些映射关系压入到Context Map中:

    (1)   parameters:这个Map中包含当前请求的请求参数

    (2)   request:包含当前请求的所有属性

    (3)   session:包含当前请求的会话的所有属性

    (4)   application:包含当前应用程序的ServletContext属性

    (5)   attr:这个Map用来按照这个顺序来检索某个属性:request、session、application

    注意:请求参数总是返回一个String类型的数组。比如我们要想知道请求参数的个数,那么正确的表达式应该是#parameters.count[0],而不是#parameters.count。

    访问Object Stack中对象的属性

    访问Object Stack里某个对象的属性,可以使用一下几种形式:

    (1)   object.propertyName

    (2)   object[propertyName]

    (3)   object[propertyName]

    另外,Object Stack里的对象还可以通过一个从零开始的下索引来引用。最顶端的对象用[0]来引用,以此类推。Strtus2中Action对象一定是位于valueStack的最顶端。

    例如:[0].propertyName  [0][propertyName]  [0][propertyName]

    Struts2中的OGNL还有个特征:如果我们指定的对象上没有找到指定的属性,那么会到指定对象的下一个对象里继续搜索,直到找到这个属性或者到达栈低。(其实现原理就是我们上面说的CompoundRoot和CompoundRootAccessor)。

    还有就是如果我们指定的属性本身也是对象,那么还可以通过同样的语法去访问这个属性对象的属性。例如:user.name.firstName

    访问Context Map上的属性

    访问Context Map上的属性的方法我们在介绍valueStack的时候已经学习过了。这里要说的是如果我们访问的属性也是对象,那么还可以通过同样的语法来访问它的属性。例如:#request[“User”][“name”]。

    调用静态属性和方法

    OGNL除了能够调用压入valueStack中的对象外,还能对任意的Java类的静态属性和方法进行调用。其表达式形式如下:

    调用静态属性: @类的全称(含包名)@静态属性名

    调用静态方法: @类的全称(含包名)@静态方法名(参数列表)

    对于压入valueStack中的对象,如果要调用其方法,直接使用object.methodName(arglist)形式进行调用.

    访问数组、List和Map类型

    访问数组和List类型的对象中对象的方法相同,都是使用从零开始的数字索引的形式,同时也可以调用数组和List类型对象上的方法。

    访问Map类型对象中的对象时需要将map中的key作为索引或者将key作为属性的方式访问,将返回与该key所对应的value

    投影与选择

    OGNL支持类似数据库中的投影(projection) 和选择(selection)。

    投影就是选出集合中每个元素的相同属性组成新的集合,类似于关系数据库的字段操作。投影操作语法为 collection.{XXX},其中XXX 是这个集合中每个元素的公共属性。

    例如:group.userList.{username}将获得某个group中的所有user的name的列表。

    选择就是过滤满足selection 条件的集合元素,类似于关系数据库的纪录操作。选择操作的语法为:collection.{X YYY},其中X 是一个选择操作符,后面则是选择用的逻辑表达式。而选择操作符有三种:

    ? 选择满足条件的所有元素

    ^ 选择满足条件的第一个元素

    $ 选择满足条件的最后一个元素

    例如:group.userList.{?#this.name != null}将获得某个group中user的name不为空的user的列表。

    创建List/Map对象

    如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中同集合相关的表达式。

    创建List:<s:set name="list"value="#{'zhangming','xiaoi','liming'}" />

    创建Map:#{ "foo" : "foo value","bar" : "bar value" }

     

    几种与OGNL有关的符号

    在Struts2中使用OGNL经常会接触到几个有关的符号:”#”,”%”,”$”。刚开始学习的时候经常分布清楚这几个符号的作用,这里我们对他们的作用大致做一个列举。

    #”的作用:

    (1)   访问非root对象的属性。例如:#session[“userName”]

    (2)   对集合进行投影与选择

    (3)   构造对象,

    %”的作用:

    在标签的属性值被理解为字符串类型时,告诉执行环境%{}里的是OGNL表达式 <s:property value="%{#foobar['foo1']}" />

     $”的作用:

    (1)   在配置文件中引用OGNL表达式(访问Action的属性)。

    (2)   在国际化资源文件中引用OGNL表达式(学习国际化时会学到)

    OGNL中的this指针

    在很多编程语言中,都有this指针的概念,它表示调用当前函数(方法)的对象。那么在OGNL中也有类似的概念。

    我们已经学过,OGNL表达式是以”.”进行串联的的一个串字符串表达式。这个表达式在被执行的时候,从左到有,每一次计算都会返回一个临时的当前对象,并在此临时对象上再次进行调用,直到执行完毕。这个临时的当前变量就存储在一个叫做this的变量中,这个this变量我们就叫它this指针。通过使用this指针,我们可以是OGNL更加灵活,更加强大。

    :使用this指针时,必须在this前面加”#”,即this指针必须以“#this”的形式出现。

    例如:group.userList.size().(#this+1).toString()

    我们可以查看ValueStack接口的实现类OgnlValueStack的源代码,会发现我们对valueStack的栈操作(pop,push,peek)实际上是对CompoundRoot类型的成员变量root的操作,而不是对Map类型的context成员变量的操作,并且root中存放的都是HashMap对象。查找的过程为现在root中进行查找,如果没有找到,那么就会在Map类型的context成员变量中进行查找。具体的实现原理需要我们去研究源代码。



  • 相关阅读:
    P4556 [Vani有约会]雨天的尾巴
    [模拟赛20180809] 旅程
    【jzoj3464】秀姿势
    【noip2013】火柴排队
    做运动
    【noip2013】花匠
    【noip2016】愤怒的小鸟
    【bzoj4326】【noip2015】运输计划
    作业二:个人编程项目——编写一个能自动生成小学四则运算题目的程序
    自我介绍
  • 原文地址:https://www.cnblogs.com/jdluojing/p/3212432.html
Copyright © 2020-2023  润新知