深入了解Mvc路由系统

时间:2024-01-21 10:24:57

请求一个MVC页面的处理过程

1.浏览器发送一个Home/Index 的链接请求到iis。iis发现时一个asp.net处理程序。则调用asp.net_isapi 扩展程序发送asp.net框架

2.在asp.net的第七个管道事件中会遍历UrlRoutingModule中RouteCollection的RoteBase集合 通过调用其GetRouteData方法进行路由匹配返回一个实现了IRoteHandler的类的对象,没有则返回null继续向下遍历

3.调用IRoteHandler的getIhttpHanderl获得实现了IhttpHandler的类的对象。再在11 到12个事件管道调用其ProcessRequset 执行处理逻辑将结果写入response

这里我们会发现很多个扩展点 1.自定义RoteBase 写入集合  2.自定义IroteHandler   3.自定义IhttpHandler

尝试自定一个RoteBase 并添加到UrlRoutingModule的RoteBase集合中

    public class MyRote : RouteBase
    {
        /// <summary>
        /// 此方法是处理请求是否跟当前路由匹配
        /// </summary>
        /// <param name="httpContext"></param>
        /// <returns></returns>
        public override RouteData GetRouteData(HttpContextBase httpContext)
        {
            //假如我们定义 如果是谷歌用户则命中路由当前路由
            )
            {
                RouteData rd= new RouteData(this, new MvcRouteHandler());
                //假设命中了此路由 解析获得请求控制器为Home action方法为Index
                rd.Values.Add("controller", "Home");
                rd.Values.Add("action", "MyRoteIndex");
                return rd;
            }
            return null;
        }

        /// <summary>
        /// 此方法 在通过Url.action()的时候根据当前路由生成路径
        /// </summary>
        /// <param name="requestContext"></param>
        /// <param name="values"></param>
        /// <returns></returns>
        public override VirtualPathData GetVirtualPath(RequestContext requestContext, RouteValueDictionary values)
        {
            return new VirtualPathData(this,"Home/MyRoteIndex");
        }
    }在RouteConfig添加到UrlRoutingModule 的RouteCollection中
public class RouteConfig
    {
        public static void RegisterRoutes(RouteCollection routes)
        {
            routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
            /***
             *  因为路由匹配是遍历匹配的 所以我们写到第一个位置。否则可能会出现匹配到下面的路由
             *添加到RoteBase集合。下面的方法是个扩展方法内部也是调用的add方法
             ***/
            routes.Add("myRote", new MyRote.MyRote());
            routes.MapRoute(
                name: "Default",
                url: "{controller}/{action}/{id}",
                defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
            );

        }
    }这个时候我们用谷歌就会发现无论我们请求哪个页面 都是调用Home/MyRoteIndex  还有在视图页面通过@url.acton 通过路由生成链接也是都是Home/MyRoteIndex

2.自己实现自己的RoteHandler 和IhttpHandler
    public class MyRoteHandler :IRouteHandler
    {
        public IHttpHandler GetHttpHandler(RequestContext requestContext)
        {
            //返回我们自己的IttpHandler处理对象
            return new MyIHttpHandler();
        }
    }

    public class MyIHttpHandler : IHttpHandler
    {

        public bool IsReusable
        {
            get { return true; }
        }

        /***
         * 我们之前说了 在第11到12个事件管道是调用这个方法写入response
         * MVC实现的IhttpHandler则是通过匹配的路由 反射调用指定控制器的action方法 再调用ActionrResult的ExcuteReuslt方法将结果写入response
         * asp.net处理页面则是调用后台页类的ProcessRequset方法处理我们程序员写的逻辑了
         ***/
        public void ProcessRequest(HttpContext context)
        {
            )
            {
                context.Response.Write("当前是谷歌浏览器");
            }
            )
            {
                context.Response.Write("当前是Ie浏览器");
            }
            else
            {
                context.Response.Write("其他浏览器");
            }
        }
    }
        /// <summary>
        /// 此方法是处理请求是否跟当前路由匹配
        /// </summary>
        /// <param name="httpContext"></param>
        /// <returns></returns>
        public override RouteData GetRouteData(HttpContextBase httpContext)
        {
            //方便调试则默认他是全部请求匹配此路由
            if (true)
            {
                //这里的RoteHandler则返回我们自己的实现
                RouteData rd= new RouteData(this, new MyRoteHandler());
                //假设命中了此路由 解析获得请求控制器为Home action方法为Index
                rd.Values.Add("controller", "Home");
                rd.Values.Add("action", "MyRoteIndex");
                return rd;
            }
        }

        /// <summary>
        /// 此方法 在通过Url.action()的时候根据当前路由生成路径
        /// </summary>
        /// <param name="requestContext"></param>
        /// <param name="values"></param>
        /// <returns></returns>
        public override VirtualPathData GetVirtualPath(RequestContext requestContext, RouteValueDictionary values)
        {
            return new VirtualPathData(this,"Home/MyRoteIndex");
        }

因为都是匹配我们的自己实现的路由 和返回我们自己的RoteHandelr 最终结果是请求任何url都是执行我们自己的IttpHanler的处理逻辑的方法