学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(一)

ASP.NET MVC是如何运行的

ASP.NET由于采用了管道式设计,所以具有很好的扩展性,整个ASP.NET MVC应用框架就是通过扩展ASP.NET实现的。通过上面对ASP.NET管道设计的介绍我们知道,ASP.NET 的扩展点主要体现在HttpModule和HttpHandler这两个核心组件之上,整个ASP.NET MVC框架就是通过自定义的HttpModule和HttpHandler建立起来的。

接下来我们通过自定义组件来模拟ASP.NET MVC的运行原理。

1.4.1建立在迷你版ASP.NET MVC上的Web应用

在介绍我们自己的ASP.NET MVC的实现原理之前,我们先来看看建立在我们自己的框架之上的Web应用如何来搭建。我们用VS建立一个空ASP.NET Web应用。

我们首先定义了如下一个SimpleModule类型,它表示最终需要绑定到View上的数据。为了验证对目标Controller和Action的解析机制,SimpleModule定义的两个属性分别表示当前请求的目标Controller和Action。为了更好的演示ASP.NET MVC的参数绑定机制(Model绑定),我们为SimpleModule定义了额外3个属性Foo、Bar和Baz,并让他们具有不同的数据类型。

public class SimpleModel
    {
        public string Controller { get; set; }
        public string Action { get; set; }
        public string Foo { get; set; }
        public int Bar { get; set; }
        public double Baz { get; set; }
    }

接下来我们定义Controller和Action,我们定义了如下一个继承自抽象类ControllerBase的HomeController。定义在HomeController中的Action方法Index具有一个SimpleModel类型的输入参数,并以ActionResult昨晚返回类型。

public class HomeController : ControllerBase
    {
        public ActionResult Index(SimpleModel model)
        {
            Action<TextWriter> callback = writer =>
            {
                writer.Write(string.Format("Controller: {0}<br/>Action: {1}<br/><br/>", model.Controller, model.Action));
                writer.Write(string.Format("Foo: {0}<br/>Bar: {1}<br/>Baz: {2}", model.Foo, model.Bar, model.Baz));
            };
            return new RawContentResult(callback);
        }
    }

方法Index返回一个RawContentResult对象。RawContentResult对象是对一个Action<TextWriter>委托的封装,他利用后者写入需要呈现的内容。在这里我们将作为参数的SimpleModel对象的两组属性(Controller/Action和Foo/Bar/Baz)的值显示出来。

ASP.NET MVC根据请求的URL来解析目标Controller的类型和Action方法名称。具体来说,我们会注册一些包含Controller和Action名称作为占位符的路由模板。如果请求地址符合相应地址模板的模式,目标Controller和Action的名称就可以正确地解析出来。我们在Global.asax中注册了如下一个模板“{Controller}/{Action}”的Route对象。除此之外,我们还注册了一个用于创建Controller对象的工厂DefaultControllerFactory。

 public class Global : System.Web.HttpApplication
    {
        protected void Application_Start(object sender, EventArgs e)
        {
            RouteTable.Routes.Add("default", new Route { Url = "{controller}/{action}" });
            ControllerBuilder.Current.SetControllerFactory(new DefaultControllerFactory());
        }
    }

路由实现了请求URL与目标Controller/Action之间的映射。ASP.NET MVC的路由建立在ASP.NET自身的路由系统之上,后者则通过自定义的HttpModule来实现。在我们的ASP.NET MVC框架中,我们将其命名为UrlRoutingModule,它与ASP.NET路由系统中对应的HttpModule类型同名,我们通过配置对自定义的HttpModule进行配置。

<system.webServer>
     <modules>
          <add name="UrlRoutingModule" type="WebApp.UrlRoutingModule, WebApp"/>
    </modules>
  </system.webServer>

这样编程和配置工作就完成了。

上面演示了如何在我们自己创建的“迷你版”ASP.NET MVC框架上搭建一个Web应用,从中可以看到它和创建一个真正的ASP.NET MVC应用别无二致。接下来我们就逐步来分析这个自定义的ASP.NET MVC框架是如何搭建起来的,他也代表了真正的ASP.NET MVC框架的基本工作原理。

总的来说,ASP.NET MVC按照这样的流程来处理响应请求:ASP.NET MVC利用路由系统对请求URL进行解析而得到目标Controller和Action的名称,以及其他相应的路由数据。它根据Controller的名称解析出目标Controller的真正类型,并将其激活(默认情况下就是根据类型以反射的机制创建Controller对象)。接下来,ASP.NET MVC利用Action名称解析出定义在目标Controller类型中对应的方法,然后执行激活Controller对象的这个方法。Action方法可以在执行过程中直接对当前请求予以响应,也可以返回一个ActionResult对象来响应请求。对于后者,ASP.NET MVC在完成目标Action方法执行之后,会执行返回的ActionResult对象来对当前请求作最终的响应。

下一章我们来讲解这个自定义的ASP.NET MVC中的路由系统的建立。

时间: 2024-10-10 07:11:29

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(一)的相关文章

学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(二)

实例演示:通过路由实现请求地址与.aspx页面的映射 我们创建一个简单的ASP.NET Web Forms应用,并采用一套独立于.aspx文件路径的URL来访问对应的Web页面,两者之间的映射通过路由来实现,我们依然沿用员工管理的场景. 首先我们将员工的所有信息(ID.姓名.性别.出生日期和所在部门)定义在如下所示的Employee类型中,然后定义一个EmployeeRepository类型来维护员工列表的数据.简单起见,员工列表通过静态字段employees表示.EmployeeReposit

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(四)

Action的执行 作为Controller的基类ControllerBase,它的Execute方法主要作用在于执行目标Action方法.如果目标Action方法返回一个ActionResult对象,它还需要执行该对象来对当前请求予以响应.在ASP.NET MVC框架中,两者的执行是通过一个叫做ActionInvoker的对象来完成的. 1.ActionInvoker 我们同样为ActionInvoker定义了一个借口IActionInvoker.如下所示.该接口定义了唯一的方法InvokeA

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(五)

2.AreaRegistration的缓存 Area的注册(主要是基于Area的路由映射注册)通过具体的AreaRegistration来完成.在应用启动的时候,ASP.NET MVC会遍历通过调用BuildManager的静态方法GetReferencedAssemblies得到的程序集列表,并从中找到所有AreaRegistration类型.如果一个应用涉及太多的程序集,则这个过程可能会耗费很多时间.为了提高性能,ASP.NET MVC会对解析出来的所有AreaRegistration类型列

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(三)

Controller的激活 ASP.NET MVC的路由系统通过注册的路由表对当前HTTP请求实施路由解析,从而得到一个用于封装路由数据的RouteData对象,这个过程是通过自定义的UrlRoutingModule对HttpApplication的PostResolveRequestCache事件进行注册实现的.由于得到的RouteData对象中已经包含了目标Controller的名称,我们需要根据该名称激活对应的Controller对象. 1.MvcRouteHandler 通过前面的介绍我

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(五)

完整的流程 对于我们创建的这个迷你版的ASP.NET MVC框架来说,虽然很多细节被直接忽略掉,但是它基本上能够展现整个ASP.NET MVC框架的全貌,支持这个开发框架的核心对象可以说一个不少.接下来我们对通过这个模拟框架展现出来的ASP.NET MVC针对请求的处理流程作一个简单的概括. 由于UrlRoutingModule这个HttpModule被注册到Web应用中,所以对于每个抵达的请求来说,当代表当前应用的HttpApplication对象的PostResolveRequestCach

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(二)

路由 对于一个ASP.NET MVC应用来说,针对HTTP请求的处理实现在目标Controller类型的某个Action,每个HTTP请求不在像ASP.NET Web Forms应用一样是针对一个物理文件,而是针对某个Controller的某个Action方法.目标Controller和Action的名称由HTTP请求的URL来决定,当ASP.NET MVC接收到抵达的请求后,其首要任务就是通过当前HTTP请求解析得到目标Controller和Action的名称,这个过程是通过ASP.NET M

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(二)

2.2.2 路由注册 ASP.NET MVC通过调用代表全局路由表的RouteCollection对象的扩展方法MapRoute进行路由注册.我们来进行一个简单的实例演示.我们依然沿用之前关于获取天气信息的路由模板,看看通过这种方式注册的Route对象针对匹配的请求将返回怎样一个RouteData对象. 我们创建一个空的ASP.NET Web程序,并手动添加"System.Web.Mvc.dll"和"System.Web.WebPages.Razor.dll"的引用

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(三)

2.2.3缺省URL参数 当通过VisualStudio的ASP.NET MVC项目模板创建一个Web应用后,它会为我们注册如下一个模板为"{controller}/{action}/{id}"的默认Route对象.3个路由模板均有相应额默认值.但是变量名为id的默认值为URLParameter.Optional.按照字面的意思,我们将其称为可缺省URL参数.那么将路由变量的默认值进行如此设置与设置一个具体的默认值有什么区别呢? routes.MapRoute( name: "

学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(六)

4.Route RouteBase是一个抽象类,在ASP.NET路由系统的应用编程接口中,Route类型是其唯一的直接继承者,在默认的情况下调用RouteCollection的MapPageRoute方法在路由表中添加的就是这么一个对象.如下面的代码片段所示,Route类型具有一个字符串类型的属性Url,它代表绑定在路由对象上的路由模板. public class Route : RouteBase { public Route(string url,IRouteHandler routeHan