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

摘要:

  虽然我们一般都是通过学习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),搭建完毕后项目结构如下:

                

  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 : Student
public 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 : College
class 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;
    }
}


(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
 *///:~


  上面的输出结果对我们来说一点也不意外,因为在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 {
    ...
}
// OgnlContext 是一个Map
public class OgnlContext extends Object implements Map {
    ...
}


  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("@[email protected](2,4)");
        Object length3 = Ognl.getValue(expression3, context, context.getRoot());
        Object expression4 = Ognl
                .parseExpression("@[email protected](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]
 *///:~

四. 使用 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
 *///:~

  由于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]
 *///:~

六. 总结

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


七. 更多

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

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

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


引用

OGNL表达式语言详解

Struts2中的OGNL详解

ognl概念和原理详解

ognl 详解

时间: 2024-10-15 15:19:51

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

Java MVC框架性能比较

- by zvane 现在各种MVC框架很多,各框架的优缺点网络上也有很多的参考文章,但介绍各框架性能方面差别的文章却不多,本人在项目开发中,感觉到采用了struts2框架的项目访问速度,明显不如原来采用了struts1框架的项目快,带着这些疑惑,我对各类MVC框架的做了一个简单的性能分析比较,其结果应该说是基本符合预期的,可供大家参考. 测试环境:CPU:酷睿2 T5750,内存:DDR2-667 2G,Web容器:Tomcat6.0,最大线程数设置为1000,操作系统:WinXP-sp3 测

8 -- 深入使用Spring -- 7...2 MVC框架与Spring整合的思考

8.7.2 MVC 框架与Spring整合的思考 对于一个基于B/S架构的JAVA EE 应用而言,用户请求总是向MVC框架的控制器请求,而当控制器拦截到用户请求后,必须调用业务逻辑组件来处理用户请求.此时有一个问题:控制器应该如何获得业务逻辑组件? 最容易想到的策略是,直接通过new 关键字创建业务逻辑组件,然后调用业务逻辑组件的方法,根据业务逻辑方法的返回值确定结果. 在实际的应用中,很少见到采用上面的访问策略,因为这是一种非常差的策略.不这样做至少有如下三个原因: ⊙ 控制器直接创建业务逻

主流MVC框架的设计模式及遵守的软件设计原则

原文地址,会不断更新  http://it.zuocheng.net/mvc-design-pattern-design-principle-summary   作程的技术博客 本文以主流的MVC框架为例,比如Java 的SSH.PHP的Symfony和Zend Framework ,在简单地剖析他们的设计原理之后,找到其中使用的设计模式:鉴赏他们的代码实现,查看设计者们都遵守了哪些软件设计原则.作此文,一为学习,二为总结.其中下面所写内容可能并不全面,也可能不准确,但会不断修改完善. 框架模式

玩转Spring MVC(二)----MVC框架

早期程序员在编写代码时是没有一个规范的,系统的业务逻辑和数据库的访问时混合在一起的,不仅增加了大量的重复工作,而且增加了后期维护的难度. 后来,人们逐渐发现某些通用的功能是可以单独的封装起来的,这样就可以减少代码量. 再后来,逐渐产生了三层架构模型,即表现层,业务逻辑层,数据层.表现层即系统的界面,控制系统展示给用户的一些东西:数据层主要实现与数据库的交互,比如数据的增删改查:业务逻辑层处于数据访问层与表示层中间,起到了数据交换中承上启下的作用,主要用于加工处理数据.通过分层,削弱了个功能层之间

Spring MVC 简述:从MVC框架普遍关注的问题说起

任何一个完备的MVC框架都需要解决Web开发过程中的一些共性的问题,比如请求的收集与分发.数据前后台流转与转换,当前最流行的SpringMVC和Struts2也不例外.本文首先概述MVC模式的分层思想与MVC框架普遍关注的问题,并以此为契机结合SpringMVC的入门级案例简要地从原理.架构角度介绍了它对这些问题的处理,包括请求处理流程.消息转换机制和数据绑定机制等核心问题.最后,本文对目前最为流行的两个MVC框架SpringMVC 和 一. MVC 模式与框架 1.MVC 模式 Java We

从MVC框架看MVC架构的设计

转自:http://blog.csdn.net/bluishglc/article/details/6690693 从MVC框架看MVC架构的设计 尽管MVC早已不是什么新鲜话题了,但是从近些年一些优秀MVC框架的设计上,我们还是会发现MVC在架构设计上的一些新亮点.本文将对传统MVC架构中的一些弊病进行解读,了解一些优秀MVC框架是如何化解这些问题的,揭示其中所折射出的设计思想与设计理念. MVC回顾   作为一种经典到不能再经典的架构模式,MVC的成功有其必然的道理,这个道理不同的人会有不同

.net的MVC框架

概述 MVC是一种思想,基于这个思想有了很多的框架,像java中的Struts,那么这个思想到底是什么呢? MVC思想 思想,也就是想法,要想知道这个想法是什么,就需要先知道这个想法因为什么产生了,要知道,没有无缘无故的恨,所以,想法也是不可能在没有任何因的情况下产生的,那么到底是什么触发了这个想法的产生,并且把它推到了高潮呢? 要想知道这个,首先,我们要分析一下在MVC真正到来之前,我们的Web编程世界的状况.当时的Web开发,主要以WebForm的开发模式,即以asp.net的事件驱动页面的

12种JavaScript MVC框架之比较

Gordon L. Hempton是西雅图的一位黑客和设计师,他花费了几个月的时间研究和比较了12种流行的JavaScript MVC框架,并在博客中总结了每种框架的优缺点,最终的结果是,Ember.js胜出. 此次比较针对的特性标准有四种,分别是: UI绑定(UI Bindings) 复合视图(Composed Views) Web表现层(Web Presentation Layer) 与其他框架良好协作(Plays Nicely with Others) 对于各种JavaScript MVC

Web开发之MVC框架

什么是MVC框架? 在这里我简要的说明一下,想要详细的了解MVC的小伙伴们可以上百度百科,或者找来相应的书籍,仔细研究和理解一下,但是要说明的的一点是,完全理解MVC并不是一件容易的事. MVC的全称为Model View Controller,是模型-视图-控制器的缩写,一种软件框架模式,通俗点讲,就是将应用程序的输入.处理和输出进行解耦,使M和V的实现代码分离,从而使同一个程序可以使用不同的表现形式. 简单来说一下MVC框架的三个核心部件的内容,即Model.View和Controller.