与MVC框架解耦的OGNL:前世今生及其基本用法

来源:互联网 发布:苹果手机越狱软件 编辑:程序博客网 时间:2024/05/21 05:58

摘要:

  虽然我们一般都是通过学习MVC框架而结缘OGNL,但它并没有与MVC框架紧紧耦合在一起,而是一个以独立的库文件出现的一种功能强大的表达式语言,也是字符串与Java对象之间沟通的桥梁。特别地,正因为它的独立性,使得我们可以十分方便地利用它构建我们自己的框架。在充分理解和掌握 OGNL 三要素 后,我们就可以通过简单一致的语法去存取Java对象树中的任意属性、调用Java对象树的方法并自动实现必要的类型转化。本文首先概述了Ognl的前世今生,并结合具体实例和源码阐述了OGNL的实质和OGNL三要素,介绍了对Java对象属性值的访问,静态、实例和构造方法的调用,对容器的访问以及对集合的操作等五个方面的介绍,奠定了学习OGNL的基础。


版权声明:

本文原创作者:书呆子Rico 
作者博客地址:http://blog.csdn.net/justloveyou_/


友情提示:

  为了更好地了解 OGNL,笔者将以两篇博客的篇幅来介绍OGNL:《与MVC框架解耦的OGNL:前世今生及其基本用法》 和 《再述OGNL:在Struts2中的应用》。其中,在本文(《与MVC框架解耦的OGNL:前世今生及其基本用法》)中,我们首先介绍了OGNL的前世今生,并结合具体实例和源码阐述了OGNL的实质、OGNL三要素和用法语法。在此基础上,本篇的姊妹篇《再述OGNL:在Struts2中的应用》详尽地介绍了OGNL的在Struts2中的应用。


一. OGNL 概述

1、OGNL的前世今生

  WebWork是建立在称为XWork的Command模式框架之上的强大的基于Web的MVC框架。关于WebWork我们大多数人可能不太熟悉,最多只是有一种在哪里见过的感觉,但是我一提Struts2,估计大家就能想起来了。众所周知,Struts2是Struts的下一代产品,是在Struts1和WebWork的技术基础上进行了合并的全新框架。需要特别注意的是,全新的Struts2的体系结构与Struts1差别巨大,因为Struts2是以WebWork为核心的,继承了更多的WebWork血统。

  实际上,WebWork 已经完全从Web层脱离出来的一个非常优秀的框架,其提供了很多核心的、Struts2还在使用的功能,包括前端拦截器(interceptor)、运行时表单属性验证、类型转换、IoC(Inversion of Control)容器等,其中就有我们今天的主角,强大的表达式语言 —— OGNL(Object Graph Notation Language)


2、OGNL 带给我们的实惠

  OGNL 是 Object Graph Navigation Language 的缩写,全称为 对象图导航语言,是一种功能强大的 表达式语言它通过简单一致的语法,可以存取Java对象树中的任意属性、调用Java对象树的方法,并能够自动实现必要的类型转化。更形象地说,如果我们把OGNL表达式看做是一个带有语义的字符串,那么OGNL无疑是这个语义字符串与Java对象之间沟通的桥梁。

  我们知道,在我们使用 MVC 架构模式进行开发Web应用时,数据往往需要在各层之间进行流转。由于数据在不同层次中的表现形式不尽相同,所以这种流转会很麻烦,特别是在Controller与View之间进行流转。实际上,数据在Controller层与View层之间流转的真正痛点就在于:数据在View层(视图页面)的表现形式是一个扁平的、不带任何数据类型的字符串,而在Controller层(Java世界)完全可以表现为一个具有丰富数据结构和数据类型的Java对象,正是由于这种数据表现形式的差异,导致我们手工执行这种转换将是一项非常复杂、低效的工作。正因为如此,为了更好地解决数据在不同层之间的数据流转问题,作为一个优秀成熟的框架,Struts2 集成了 WebWork 中的 OGNL 来帮助我们解决个问题。因此,当我们在使用Struts2时,会发现OGNL充斥在前后台数据传递与存储的方方面面,也给我们带来了极大的方便。


  Ps:更多关于 MVC框架数据流转问题与OGNL在Web中的魅力 等内容请读者移步本文的姊妹篇《再述OGNL:在Struts2中的应用》。


3、小结

  OGNL是模板语言的一个重要补充,对表现层技术而言是一次里程碑式的进步。在我们常见的视图组件,包括 Jsp 2.0,Velocity,Jelly 等,虽然都有类似的功能,比如,在 Jsp 2.0 中我们可以使用其提供的 EL 表达式完成类似的功能。但是,OGNL比它们要完善的多得多,而且以一个独立的库文件出现,十分方便我们构建自己的框架。


二. OGNL 深度解读:从一个例子说起

  我们在上文已经提到,OGNL 以一个独立的库文件出现,十分方便我们构建自己的框架。那么,我们首先新建一个Java Project,然后从Struts2的相关依赖包中导入 ognl-x.x.xx.jar (本人使用的struts-2.1.6中的ognl-2.6.11.jar),搭建完毕后项目结构如下:

                Image 4.png-8.8kB

  Ps:若读者不知如何重现笔者获取OGNL的jar包的过程,请笔者移步我的博文《Struts2 实战:从 登录Demo 看 Struts2 应用开发》去详细了解其获取过程,此不赘述。另外,由于OGNL是Apache开源项目的子项目,所以我们可以从 Apache Commons 下载OGNL的jar包。


1、OGNL应用实例

  上述的Java Project包含两个JavaBean类和一个OGNL测试类,我们将围绕这个Project展开对OGNL的介绍。我们先看一下该Project中各个类的源码:


(1). 两个JavaBean

package cn.tju.edu.rico.test;import java.util.HashSet;import java.util.Set;// Java Bean : Studentpublic class Student {    private College College;      private String name;      private String gentle;      private double height;    private int age;     // 无参构造器    public Student() {      }      public Student(String name, int age, double height) {        super();        this.name = name;        this.height = height;        this.age = age;    }    //getter & setter    public College getCollege() {          return College;      }      public void setCollege(College College) {          this.College = College;      }      public String getName() {          return name;      }      public void setName(String name) {          this.name = name;      }    public String getGentle() {        return gentle;    }    public void setGentle(String gentle) {        this.gentle = gentle;    }    public double getHeight() {        return height;    }    public void setHeight(double height) {        this.height = height;    }    public int getAge() {        return age;    }    public void setAge(int age) {        this.age = age;    }    @Override    public String toString() {        return "Student [name=" + name + ", height=" + height + ", age=" + age                + "]";    } }// Java Bean : Collegeclass College {    private String name;      private Set<Student> Students = new HashSet<Student>();      // 无参构造器    public College() {      }      //getter & setter    public String getName() {          return name;      }      public void setName(String name) {          this.name = name;      }      public Set<Student> getStudents() {          return Students;      }      public void setStudents(Set<Student> Students) {          this.Students = Students;      }  }
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104

(2). OGNL测试类

package cn.tju.edu.rico.test;import ognl.Ognl;import ognl.OgnlContext;import ognl.OgnlException;public class OGNLTest {    public static void main(String[] args) throws OgnlException {        // 新建一个学校对象        College college = new College();        college.setName("TJU");        // 新建一个学生对象        Student stu = new Student();        stu.setName("Rico");        // 构建一个OgnlContext对象 ,并将上述学校、学生对象放入Ognl上下文环境(本质是一个Map)中        OgnlContext context = new OgnlContext();        context.put("college", college);        context.put("stu", stu);        // 将学生设置为根对象        context.setRoot(stu);        // 构建Ognl表达式的树状表示        Object expression1 = Ognl.parseExpression("#college.name");        Object expression2 = Ognl.parseExpression("name");        Object expression3 = Ognl.parseExpression("#stu.name");        // 根据Ognl表达式给Java对象设置值,将TJU改为NEU        Ognl.setValue(expression1, context, context.getRoot(), "NEU");        // 根据Ognl表达式获取Java对象的(属性)值        Object collegeName = Ognl.getValue(expression1, context,                context.getRoot());        Object stuName2 = Ognl                .getValue(expression2, context, context.getRoot());        Object stuName3 = Ognl                .getValue(expression3, context, context.getRoot());        System.out.println(collegeName);        System.out.println(stuName2);        System.out.println(stuName3);    }}/* Output:         NEU        Rico        Rico *///:~
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51

  上面的输出结果对我们来说一点也不意外,因为在Struts2中,我们常常使用上述方式访问Stack Context(Action Context)及其根对象Value Stack。根据这个例子我们也能够看出,所谓的对象图导航语言本质上就是通过 类似”放置到OgnlContext中的名字.属性名字” 的方式去获取对应对象的属性值。特别的是,对于根对象的属性的访问,我们只需直接利用属性名字访问即可,因为根对象只有一个,OGNL会默认从OgnlContext中的根对象中去寻找;而对于普通对象的属性的访问,我们使用类似”#放置到OgnlContext中的名字.属性名字”的方式去访问,这时OGNL在解析表达式的时候发现表达式开头带有”#”,就会去普通对象中去寻找。当然,使用这种方式也可以访问根对象的属性,但是若在访问普通对象时不加前缀“#”,将会抛出 ognl.OgnlException。


2、OGNL 三要素

  事实上,OGNL表达式的计算是围绕OGNL上下文(OgnlContext)进行的,而OGNL上下文实际上就是一个Map对象。我们从上述的例子可以看出,无论是setValue方法还是getValue方法,它们均包含三个核心参数,即 tree(OGNL表达式)context(Ognl上下文)root(Ognl上下文的根对象),我们将其称之为 OGNL的三要素,因为 OGNL 的操作实际上就是围绕着这三个参数而进行的。这三者的关系如下图所示:

            这里写图片描述


(1).表达式(Expression)

  表达式是整个OGNL的核心,所有的OGNL操作都是对表达式解析后进行的。准确的来说,表达式表达了此OGNL操作的语义,即表明了此OGNL操作“要干什么”


(2).上下文环境(Context)

  我们在上文提到,所有的OGNL操作都是在一个特定的环境中进行的,这个环境就是OGNL的上下文环境(OGNL Context)。更直白地说,OGNL上下文为OGNL表达式的提供了具体的运行环境。需要指出的是,我们完全可以像操作Map那样将一些数据设置到OGNL Context中,以便我们通过OGNL访问。准确的来说,Context为OGNL表达式提供了具体环境,为OGNL操作“提供支持”


(3).根对象(Root Object)

  根对象是OGNL Context中的一员,并且整个OGNL Context最多只允许有一个根对象。也就是说,OGNL Context中共有两类对象,即 根对象  普通对象 ,它们的差异具体表现在访问方式上,我们针对根对象的存取操作的表达式不需要增加任何前缀(下文会具体提到)。根对象从侧面指明了OGNL操作所针对的对象类别,也就是说,在表达式规定了“干什么”之后,根对象指明了我们到底“对谁干”(根对象还是普通对象)。


3、OGNL源码解读

  在上述的例子中,无论是setValue方法还是getValue方法,都是ognl.Ognl类提供的两个静态方法。事实上,在OGNL中,我们最常用到的两个类是 ognl.Ognl  ognl.OgnlContext ognl.Ognl类是一个抽象类,并提供了一系列用于解析和解释执行Ognl表达式的方法,而抽象类则是专门用来继承的;ognl.OgnlContext类则为Ognl表达式提供了一个执行环境,这个类实现了Map接口,所以允许我们通过使用Map的put(key,value)方法向OgnlContext添加各种类型的对象。需要注意的是,在OgnlContext中一共有两种对象,第一种是根对象,根对象在整个OgnlContext中有且最多只能有一个,我们可以通过调用OgnlContext.setRoot(obj)设置根对象。另外一种就是普通对象,它的个数不受限制。它们最重要的一个区别是在对象属性的获取方式上,前者可直接访问,后者需使用类似”#放置到OgnlContext中的名字.属性名字”的方式去访问 。下面给出了ognl.Ognl 与 ognl.OgnlContext 的声明方式,关于它们更多的细节本文不在赘述,读者若想进一步了解,请自行阅读源码。

// Ognl 是一个抽象类,而抽象类则是专门用来继承的public abstract class Ognl {    ...}
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4
// OgnlContext 是一个Mappublic class OgnlContext extends Object implements Map {    ...}
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

  Ps:关于 抽象类的深层次表述 请读者移步我的博文《Java 的抽象特性:抽象类与接口深度解析》。


4、小结

  到此为止,我相信通过上面的例子和表述,我们对Ognl表达式有了一个更深入的了解和认识。此外,我们知道对于普通对象的属性的访问,我们只能使用类似”#放置到OgnlContext中的名字.属性名字”的方式去访问,而对于根对象的属性的访问,我们可以通过以下两种方式去访问:

  • 直接利用属性名字访问;

  • 类似”#放置到OgnlContext中的名字.属性名字”的方式去访问; 
      
       
    下文我们将着重讲述Ognl的基本用法,抛开MVC框架单独了解它的用法便于我们进一步理解Ognl在Struts2中的使用方式。


三. 使用 OGNL 去访问方法

  我们除了利用Ognl表达式访问对象的属性,还可以使用它来访问方法。当然,对于方法的访问,又可以分为 对静态方法的访问对实例方法的访问 和 对构造方法的访问,我们先看下面的例子:

package cn.tju.edu.rico.test;import ognl.Ognl;import ognl.OgnlContext;import ognl.OgnlException;public class OGNLTest2 {    public static void main(String[] args) throws OgnlException {        // 新建一个学校对象        College college = new College();        college.setName("NEU");        // 新建一个学生对象        Student stu = new Student();        stu.setName("Livia");        stu.setCollege(college);        stu.setGentle("boy");        // 构建一个OgnlContext对象,并将上述学校和学生对象放入Ognl上下文环境中        OgnlContext context = new OgnlContext();        context.put("college", college);        context.put("stu", stu);        // 将学生对象设置为根对象        context.setRoot(stu);        // 访问实例方法        Object expression1 = Ognl.parseExpression("getGentle()");        Object length1 = Ognl.getValue(expression1, context, context.getRoot());        Object expression2 = Ognl.parseExpression("#college.name.length()");        Object length2 = Ognl.getValue(expression2, context, context.getRoot());        System.out.println(length1);        System.out.println(length2);        // 访问静态方法        Object expression3 = Ognl.parseExpression("@java.lang.Math@max(2,4)");        Object length3 = Ognl.getValue(expression3, context, context.getRoot());        Object expression4 = Ognl                .parseExpression("@java.lang.String@valueOf(name.length())");        Object length4 = Ognl.getValue(expression4, context, context.getRoot());        System.out.println(length3);        System.out.println(length4);        // 访问构造方法:通过Ognl表达式构建一个LinkedList对象,注意使用全类名        Object expression5 = Ognl.parseExpression("new java.util.LinkedList()");        List list = (List)Ognl.getValue(expression5, context, context.getRoot());        list.add("list");        list.add("rico");         System.out.println(list);    }}/* Output:         boy        3        4        5        [list, rico] *///:~
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58

四. 使用 OGNL 去访问容器对象

   我们还可以利用Ognl表达式访问容器对象,包括数组,List,Set,Map等,看下面的例子:

package cn.tju.edu.rico.test;import java.util.ArrayList;import java.util.HashMap;import java.util.List;import java.util.Map;import ognl.Ognl;import ognl.OgnlContext;import ognl.OgnlException;public class OGNLTest3 {    public static void main(String[] args) throws OgnlException {        OgnlContext context = new OgnlContext();        // 处理数组类型        String[] strs = new String[] { "a", "b", "c" };        context.put("strs", strs);        System.out.println(Ognl.getValue("#strs[2]", context, context.getRoot()));        // 处理List类型        List<String> words = new ArrayList<String>();        words.add("rico");        words.add("livia");        words.add("neu");        context.put("words", words);        System.out.println(Ognl.getValue("#words[0]", context,context.getRoot()));        // 处理Map类型        Map<String,String> map = new HashMap<String,String>();        map.put("ad", "d");        map.put("Rico", "China");        map.put("campus", "neu");        context.put("map", map);        System.out.println(Ognl.getValue("#map['Rico']",context, context.getRoot()));        // 处理Set类型:由于Set的无序性,所以不能通过这种访问Set(只能迭代输出),会抛出 ognl.NoSuchPropertyException;        Set<String> set = new HashSet<String>();        set.add("rico");        set.add("livia");        set.add("neu");        context.put("set", set);        System.out.println(Ognl.getValue("#set[2]", context, context.getRoot()));    }}/* Output:         c        rico        China        Exception in thread "main" ognl.NoSuchPropertyException: java.util.HashSet.2 *///:~
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55

  由于Set是无序的且没有索引,所以我们只能对其进行迭代输出。Struts2 提供了一组逻辑控制标签,其中就有iterator,它可以完美完成这件事情。关于Struts2 的逻辑控制标签的叙述详见本文的姊妹篇《再述OGNL:在Struts2中的应用》。

  由于HashSet就是通过HashMap实现的,所以更多关于 HashSet的了解,读者可以参考我的博文《Map 综述(一):彻头彻尾理解 HashMap》。


五. 使用 OGNL 对容器进行操作

  我们还可以利用Ognl表达式对容器对象作一些操作,比如 过滤 和 投影过滤指的是将原集合中不符合条件的对象过滤掉,然后将满足条件的对象,构建一个新的集合对象返回,Ognl过滤表达式的写法是:collection.{?|^|$ expression};投影指的是将原集合中所有对象的某个属性抽取出来,单独构成一个新的集合对象返回,基础语法为 :collection.{expression}。特别需要注意的是,无论是过滤操作还是投影操作,它们的操作对象和操作结果都是一个容器。

package cn.tju.edu.rico.test;import java.util.ArrayList;import java.util.Collections;import java.util.List;import ognl.Ognl;import ognl.OgnlContext;import ognl.OgnlException;public class OGNLTest4 {    public static void main(String[] args) throws OgnlException {          Student s1 = new Student("Tom", 22, 170.3);          Student s2 = new Student("Jack", 21, 176.2);          Student s3 = new Student("Tomas", 23, 180.1);          Student s4 = new Student("Lucy", 20, 163.3);          List<Student> stus = new ArrayList<Student>();          Collections.addAll(stus, s1, s2, s3, s4);          // 新建OgnlContext对象          OgnlContext context = new OgnlContext();          context.put("stus", stus);          // 过滤(filtering),collection.{? expression}          // 利用过滤获取身高在175以上的所有学生集合          // 输出结果:[Student [name=Jack, age=21, height=176.2], Student [name=Tomas, age=23, height=180.1]]          System.out.println(Ognl.getValue("#stus.{? #this.height > 175.0}", context, context.getRoot()));          // 过滤(filtering),collection.{^ expression}          // 利用过滤获取身高在175以上的所有学生集合中第一个元素          // 输出结果:[Student [name=Jack, age=21, height=176.2]]          System.out.println(Ognl.getValue("#stus.{^ #this.height > 175.0}", context, context.getRoot()));          // 过滤(filtering),collection.{$ expression}          // 利用过滤获取身高在175以上的所有学生集合的最后一个元素          // 输出结果:[Student [name=Tomas, age=23, height=180.1]]          System.out.println(Ognl.getValue("#stus.{$ #this.height > 175.0}", context, context.getRoot()));          // 投影(projection), collection. {expression}          // 获取集合中的所有学生的姓名          // 输出结果:[Tom, Jack, Tomas, Lucy]          System.out.println(Ognl.getValue("#stus.{name}", context, context.getRoot()));      }  }/* Output:         [Student [name=Jack, height=176.2, age=21], Student [name=Tomas, height=180.1, age=23]]        [Student [name=Jack, height=176.2, age=21]]        [Student [name=Tomas, height=180.1, age=23]]        [Tom, Jack, Tomas, Lucy] *///:~
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51

六. 总结

  虽然我们一般都是通过学习MVC框架而结缘OGNL,但它并没有与MVC框架紧紧耦合在一起,而是一个以独立的库文件出现的一种功能强大的表达式语言,也是字符串与Java对象之间沟通的桥梁。特别地,正因为它的独立性,使得我们可以十分方便地利用它构建我们自己的框架。在充分理解和掌握 OGNL 三要素 后,我们就可以通过简单一致的语法去存取Java对象树中的任意属性、调用Java对象树的方法并自动实现必要的类型转化。本文首先概述了Ognl的前世今生,并结合具体实例和源码阐述了OGNL的实质和OGNL三要素,介绍了对Java对象属性值的访问,静态、实例和构造方法的调用,对容器的访问以及对集合的操作等五个方面的介绍,奠定了学习OGNL的基础。


七. 更多

  更多关于 MVC框架数据流转问题与OGNL在Web中的魅力 等内容请读者移步本文的姊妹篇《再述OGNL:在Struts2中的应用》。

  更多关于 抽象类的深层次表述 请读者移步我的博文《Java 的抽象特性:抽象类与接口深度解析》。

  由于HashSet就是通过HashMap实现的,所以更多关于 HashSet的了解,读者可以参考我的博文《Map 综述(一):彻头彻尾理解 HashMap》。

0 0
原创粉丝点击