系列导航地址http://www.cnblogs.com/fzrain/p/3490137.htmlhtml
题外话:因为这个技术点是新学的,并不属于原系列,但借助了原系列的项目背景,故命名外传系列,之后也可能在这个系列中附加一些新的技术。git
在Web Api 2.0中,提出了一种新的配置路由方式——基于特性的路由(Attribute-based Routing),在咱们以前介绍的配置路由方式称为——基于公约的路由(Convention-based Routing),新的路由配置方式一样应用在MVC5中,所以本文就来介绍一下基于特性的路由。github
在以前的一篇文章中,咱们处理了这么一个业务——实现学生选课。咱们是经过在“WebApiConfig”定制了一条路由数据来实现的,这条路由实现了选课以及根据课程Id来查询选择选择该课程的全部学生信息,感受设计还能够。在实际应用中通常来讲查询用的是最多的,使用Attribute Routing来注册路由会更灵活,控制起来也更方便,更符合Rest。本文以根据课程的名字来查询全部选择该课程的学生信息以及根据课程名字以及学生名字查询某一学生的信息。web
顾名思义,新路由将使用一个特性来实现路由注册,以下:api
[AttributeUsage(AttributeTargets.Method | AttributeTargets.Class, AllowMultiple=true, Inherited=true)] public sealed class RouteAttribute : Attribute, IHttpRouteInfoProvider { public RouteAttribute(); public RouteAttribute(string template); public string Name { get; set; } public int Order { get; set; } public string Template { get; private set; } }
这个类包含3个属性:Name指路由的名字,Order是指路由的顺序,Template就是咱们要去匹配URL的模板app
原理性的东西很少介绍的(我也没多研究 呵呵),能把学到的技术运用到实际中才是王道,上来一大堆原理容易晕,之后须要深刻研究再看原理。ide
在“EnrollmentsController”中新增一个方法GetStudentsInfo:测试
[Route("api/enrollments/{courseName}/{studentName?}")] public IEnumerable<StudentBaseModel> GetStudentsInfo(string courseName, string studentName="") { IQueryable<Student> query; Course course= TheRepository.GetAllCourses().Where(c => c.Name == courseName).FirstOrDefault(); if (course==null ) { return null ; } query = TheRepository.GetEnrolledStudentsInCourse(course.Id).OrderBy(s => s.LastName); if (!string.IsNullOrWhiteSpace(studentName)) { query = query.Where(s => s.FirstName == studentName); } var totalCount = query.Count(); System.Web.HttpContext.Current.Response.Headers.Add("X-InlineCount", totalCount.ToString()); var results = query .ToList() .Select(s => TheModelFactory.Create(s)); return results; }
在咱们的Action上使用了RouteAttribute。分析一下这个URL模板("api/enrollments/{courseName}/{studentName?}"),{courseName}会匹配到Action的courseName参数上,对于Action的另外一个参数studentName是一个可选参数,也就是说请求中没有给出值那么就是默认的空字符串,有值得话就会被赋值,所以咱们在“{studentName}”后面加上? 标记为可选的URI参数。spa
ok,就这么简单,测试一次:设计
结果:
呃,出错了。。。
出错了,不过无论怎么说解决方案总归是有的,首先看下错误缘由:是在LearningControllerSelector类的方法中出现了空引用,那么咱们就不得不看下这个方法:
public override HttpControllerDescriptor SelectController(HttpRequestMessage request) { var controllers = GetControllerMapping(); //Will ignore any controls in same name even if they are in different namepsace var routeData = request.GetRouteData(); var controllerName = routeData.Values["controller"].ToString(); HttpControllerDescriptor controllerDescriptor; if (controllers.TryGetValue(controllerName, out controllerDescriptor)) { var version = "2"; var versionedControllerName = string.Concat(controllerName, "V", version); HttpControllerDescriptor versionedControllerDescriptor; if (controllers.TryGetValue(versionedControllerName, out versionedControllerDescriptor)) { return versionedControllerDescriptor; } return controllerDescriptor; } return null; }
看了这段代码咱们不难发现这个方法是咱们重写了基类的方法,目的是为了实现现版本控制(详情可移步:http://www.cnblogs.com/fzrain/p/3558765.html),在咱们重写的方法中咱们用到了RouteData中包含的Controller的名字,而这个名字是由基于公约的路由与URI匹配获得的,所以咱们这里确定是没有的。由于咱们的项目混合了2钟路由配置,而自定义的方法是针对基于公约的路由配置,所以对于Attribute Routing咱们直接使用默认的选择方式:
在“var controllerName = routeData.Values["controller"].ToString(); ”代码以前加一个判断:
if (string.IsNullOrWhiteSpace(routeData.Route.RouteTemplate)) { return base.SelectController(request); }
再次请求就会获得以下结果:
结果正确,解决方案感受还能够再考虑考虑,你们有什么好的建议能够提。可是无论怎么说,有一点是明确的:对于基于特性的路由规则,不须要包含Controller的名字,在本案列中感受上是寻找Acton上的路由模板来匹配请求的URI,而后肯定对应的Action。
基于特性路由(Attribute Routing)是一种新的定制路由规则方式,与传统的配置方式相比更灵活,但同时比较分散,在web api中,2钟路由配置方式能够共存,因此本文只是介绍一种新的配置方式,作项目时也能够多一种选择。
本文参考相关连接:http://www.cnblogs.com/aehyok/p/3449851.html