`
wyf
  • 浏览: 436625 次
  • 性别: Icon_minigender_1
  • 来自: 唐山
社区版块
存档分类
最新评论

ASP.NET MVC5 新特性:Attribute路由使用详解

    博客分类:
  • MVC
 
阅读更多
1、什么是Attribute路由?怎么样启用Attribute路由?

  微软在 ASP.NET MVC5 中引入了一种新型路由:Attribute路由,顾名思义,Attribute路由是通过Attribute来定义路由。当然,MVC5也支持以前定义路由的方式,你可以在一个项目中混合使用这两种方式来定义路由。

  在以前的版本中我们通常在 RouteConfig.cs 文件中通过以下方式来定义路由:

routes.MapRoute(

name: "ProductPage",

url: "{productId}/{productTitle}",

defaults: new { controller = "Products", action = "Show" },

constraints: new { productId = "\\d+" }

);

  在MVC5中,我们可以把路由定义和 Action 放在一起:

[Route("{productId:int}/{productTitle}")]

public ActionResult Show(int productId) { ... }

  当然,首先得启用Attribute路由,我们可以调用MapMvcAttributeRoutes方法来启用Attribute路由:

public class RouteConfig

{

public static void RegisterRoutes(RouteCollection routes)

{

routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

routes.MapMvcAttributeRoutes();

}

}


2、URL可选参数和默认值

  我们可以使用问号“?”来标记一个可选参数,也可以对参数设定默认值:

public class BooksController : Controller

{

// 匹配: /books

// 匹配: /books/1430210079

// 问号表示 isbn 是可选的

[Route("books/{isbn?}")]

public ActionResult View(string isbn)

{

if (!String.IsNullOrEmpty(isbn))

{

return View("OneBook", GetBook(isbn));

}

return View("AllBooks", GetBooks());

}

// 匹配: /books/lang

// 匹配: /books/lang/en

// 匹配: /books/lang/he

// 如果URL不传递 lang 参数,则lang的值为“en”

[Route("books/lang/{lang=en}")]

public ActionResult ViewByLanguage(string lang)

{

return View("OneBook", GetBooksByLanguage(lang));

}

}


3、路由前缀

  有时候在同一个 Controller 中,所有 Action 匹配的 URL 都拥有相同的前缀,如下:

public class ReviewsController : Controller

{

// 匹配: /reviews

[Route("reviews")]

public ActionResult Index() { ... }

// 匹配: /reviews/5

[Route("reviews/{reviewId}")]

public ActionResult Show(int reviewId) { ... }

// 匹配: /reviews/5/edit

[Route("reviews/{reviewId}/edit")]

public ActionResult Edit(int reviewId) { ... }

}

  我们看到 ReviewsController 下的所有 Action 前面都带有 "reviews",这时我们可以在 Controller 上使用 [RoutePrefix]设置路由前缀,为每个 Action 所匹配的 URL 加上共同的前缀 "reviews":

[RoutePrefix("reviews")]

public class ReviewsController : Controller

{

// 匹配: /reviews

[Route]

public ActionResult Index() { ... }

// 匹配: /reviews/5

[Route("{reviewId}")]

public ActionResult Show(int reviewId) { ... }

// 匹配: /reviews/5/edit

[Route("{reviewId}/edit")]

public ActionResult Edit(int reviewId) { ... }

}

  但是,如果某一个 Action 不想要这个前缀怎么办?当然有办法,我们可以用波浪号“~”来去掉它:

[RoutePrefix("reviews")]

public class ReviewsController : Controller

{

// 匹配: /spotlight-review

[Route("~/spotlight-review")]

public ActionResult ShowSpotlight() { ... }

...

}


4、默认路由

  我们除了可以在 Action 上使用[Route]外,也可以用在 Controller 上,当[Route]用在 Controller 上时,它就定义了一个默认路由规则,它会对这个 Controller 下的所有 Action 起作用,除非某个 Action 上也应用了 [Route] 特性覆盖了 Controller 上的[Route]。但要注意的是应用在 Controller 上的 [Route] 一定要加上 {action},否则会抛出“RouteData 必须包含名为'action'且值为非空字符串的项。”错误。应用在 Action 上的 [Route] 则不用加,因为 {action} 就是当前 Action。

[RoutePrefix("promotions")]

[Route("{action=index}")]

//上面定义了默认路由,并且{action}的默认值为"index",

//也就是说 URL 不包含 {action} 时,默认调用的 Action 是 Index。

public class ReviewsController : Controller

{

// 匹配: /promotions

public ActionResult Index() { ... }

// 匹配: /promotions/archive

public ActionResult Archive() { ... }

// 匹配: /promotions/new

public ActionResult New() { ... }

// 匹配: /promotions/edit/5

// 这里覆盖了默认路由规则

// 按照默认路由,这里应该匹配:/promotions/editProduct?promoId=5

[Route("edit/{promoId:int}")]

public ActionResult EditProduct(int promoId) { ... }

}


5、路由约束

  路由约束可以让你指定参数的类型以及范围等,格式为:{参数:约束},举例如下:

// 匹配: /users/5

[Route("users/{id:int}"]

// 这里约束了参数“id”必须为整数类型

public ActionResult GetUserById(int id) { ... }


  下面是支持的路由约束列表:

alpha,必须为大小写字母(a-z,A-Z),如:{x:alpha};

bool,必须为布尔值,如:{x:bool}

datetime,必须为DateTime(时间和日期)类型,如:{x:datetime}

decimal,必须为decimal类型,如:{x:decimal}

double,必须为64bit浮点数,如:{x:double}

float,必须为32bit浮点数,如:{x:float}

guid,必须为GUID,如:{x:guid}

int,必须为32bit整数,如:{x:int}

length,字符串长度必须为指定值或者在指定范围内,如:{x:length(6)} {x:length(1,20)}

long,必须为64bit整数,如:{x:long}

max,小于等于指定值的整数,如:{x:max(10)}

maxlength,字符串长度小于等于指定值,如:{x:maxlength(10)}

min,大于等于指定值的整数整数,如:{x:min(10)}

minlength,字符串长度大于等于指定值,如:{x:minlength(10)}

range,必须是给定范围内的整数,如:{x:range(10,50)}

regex,必须与正则表达式匹配,如:{x:(^\d{3}-\d{3}-\d{4}$)}


  你可以在一个参数后面应用多个约束,用冒号分隔它们,如下:

// 匹配: /users/5

// 但是不匹配 /users/10000000000 因为id的值已经超过了int.MaxValue,

// 也不匹配 /users/0 因为后面有个min(1)约束,id 的值必须大于等于 1.

[Route("users/{id:int:min(1)}")]

public ActionResult GetUserById(int id) { ... }


  值得注意的是加在可选参数上的约束,例如:

// 匹配: /greetings/bye

// 也匹配 /greetings 因为message是可选参数,

// 但是不匹配 /greetings/see-you-tomorrow 因为有maxlength(3)约束.

[Route("greetings/{message:maxlength(3)?}")]

public ActionResult Greet(string message) { ... }


6、自定义路由约束

  我们可以通过实现 IRouteConstraint 接口来自定义路由约束。下面的例子展示如何自定义路由约束:

public class ValuesConstraint : IRouteConstraint

{

private readonly string[] validOptions;

public ValuesConstraint(string options)

{

validOptions = options.Split('|');

}

public bool Match(HttpContextBase httpContext, Route route,

string parameterName, RouteValueDictionary values,

RouteDirection routeDirection)

{

object value;

if (values.TryGetValue(parameterName, out value) && value != null)

{

return validOptions.Contains(value.ToString(), StringComparer.OrdinalIgnoreCase);

}

return false;

}

}

下面的代码是显示如何注册自定义的路由约束

public class RouteConfig

{

public static void RegisterRoutes(RouteCollection routes)

{

routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

var constraintsResolver = new DefaultInlineConstraintResolver();

constraintsResolver.ConstraintMap.Add("values", typeof(ValuesConstraint));

routes.MapMvcAttributeRoutes(constraintsResolver);

}

}

现在我们就可以在代码中使用这些自定义的路由约束了:

public class TemperatureController : Controller

{

// 匹配 temp/celsius 以及 /temp/fahrenheit 但不匹配 /temp/kelvin

[Route("temp/{scale:values(celsius|fahrenheit)}")]

public ActionResult Show(string scale)

{

return Content("scale is " + scale);

}

}


7、路由名称

  你可以为路由规则指定一个名称,以方便生成相应的URL,举例如下:

[Route("menu", Name = "mainmenu")]

public ActionResult MainMenu() { ... }

  你可以使用 Url.RouteUrl 来生成相应的 URL:

<a href="@Url.RouteUrl("mainmenu")">Main menu</a>


8、Area

  ASP.NET MVC 的 Area 概念对组织大型Web应用程序很有帮助,在Attribute路由中当然少不了对它的支持,只要使用 [RouteArea],就可以把 Controller 归属到某一个 Area 下,这时你可以放心的删除 Area 下的 AreaRegistration 类了:


[RouteArea("Admin")]

[RoutePrefix("menu")]

[Route("{action}")]

public class MenuController : Controller

{

// 匹配: /admin/menu/login

public ActionResult Login() { ... }

// 匹配: /admin/menu/show-options

[Route("show-options")]

public ActionResult Options() { ... }

// 匹配: /stats

[Route("~/stats")]

public ActionResult Stats() { ... }

}


  现在你可以和以往的版本一样使用 "Admin" Area,下面的代码会生成 URL "/Admin/menu/show-options":

Url.Action("Options", "Menu", new { Area = "Admin" })

  你还可以通过 AreaPrefix 来设置 Area 前缀,例如:

[RouteArea("BackOffice", AreaPrefix = "back-office")]


  如果你同时使用 Attribute、AreaRegistration 类这两种方式来注册 Area 的话,你应该在注册 Attribute 路由和传统路由映射之间使用 AreaRegistration 注册 Area,原因很简单,路由注册顺序必须是从最精确的匹配规则开始再到普通的匹配规则,最后才是模糊的匹配规则,这样就避免了在进行路由匹配时,过早的匹配了模糊规则,而相对精确的匹配起不到任何作用。下面的例子展示了这一点:


public static void RegisterRoutes(RouteCollection routes)

{

routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

routes.MapMvcAttributeRoutes();

AreaRegistration.RegisterAllAreas();

routes.MapRoute(

name: "Default",

url: "{controller}/{action}/{id}",

defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }

);

}

 

分享到:
评论

相关推荐

    ASP.NET MVC5 新特性

    ### ASP.NET MVC5 新特性详解 #### 一、Attribute路由 **Attribute路由**是ASP.NET MVC5中的一个重要新特性,它允许开发者通过属性的方式定义路由规则,与传统的在`RouteConfig.cs`文件中集中配置路由相比,...

    精通ASP.NET MVC 5.pdf

    4. **路由(Routing)**:ASP.NET MVC 5使用URL路由系统,允许开发者自定义URL结构,使URL更具语义化。路由配置在Global.asax文件中,通过RegisterRoutes方法定义。 5. **身份认证与授权(Authentication & ...

    ASP.NET MVC5增删改查

    ASP.NET MVC5增删改查,包含LINQ和ADO两种(ADO只实现了List Detail和Delete,Edit留给你自己思考了),LINQ版本实现了完整的CRUD,并且演示了MVC5 的新特性Attribute Route以及异步的修改、删除控制器。 相關討論參...

    ASP.NET MVC5项目实现增删改查,供新手学习

    ASP.NET MVC5增删改查,包含LINQ和ADO两种(ADO只实现了List Detail和Delete,Edit留给你自己思考了),LINQ版本实现了完整的CRUD,并且演示了MVC5 的新特性Attribute Route以及异步的修改、删除控制器。

    ASP.NET MVC3完美安装包(含升级包)

    3. 异步控制器:ASP.NET MVC3支持异步操作,可以处理长时间运行的任务,提高服务器的响应速度和并发性能。 4. 增强的模型绑定:模型绑定可以自动将HTTP请求数据绑定到控制器的行动参数,简化了数据接收和验证的过程...

    【ASP.NET编程知识】asp.net MVC使用PagedList.MVC实现分页效果.docx

    ASP.NET MVC 使用 PagedList.MVC 实现分页效果 ASP.NET MVC 是一个基于模型视图控制器(MVC)架构的Web应用程序框架,它提供了一个灵活的方法来构建Web应用程序。PagedList.MVC 是一个 ASP.NET MVC 的分页插件,它...

    ASP.NET MVC 5 with Bootstrap and Knockout

    第11章使用Attribute定义URL路由 Attribute路由基础知识 路由前缀 路由约束 小结 第12章胖模型、瘦控制器 关注点分离 服务与行为 小结 第四部分应用实例 第13章构建购物车 购物车需求 购物车项目 ...

    asp.net mvc4/5 plugin

    ASP.NET MVC4/5 插件开发详解 ASP.NET MVC 是一种基于模型-视图-控制器(Model-View-Controller)的设计模式的Web应用程序框架,由微软开发,它结合了ASP.NET Web Forms的事件驱动模式和ASP.NET AJAX的异步交互能力...

    ASP.NET MVC4 Web 中文版.zip

    1. **改进的路由系统**:ASP.NET MVC4引入了基于特性(Attribute Routing)的路由,允许开发者通过在控制器和操作方法上添加属性来定义路由规则,使路由配置更加灵活。 2. **移动支持**:ASP.NET MVC4包含了一套...

    ASP.NET+MVC下基于RBAC权限认证的设计与实现

    在探讨ASP.NET+MVC下基于RBAC(Role-Based Access Control)权限认证的设计与实现时,我们首先要理解几个核心概念:ASP.NET MVC框架、RBAC模型以及它们在现代Web应用程序中的应用。 ### ASP.NET MVC框架 ASP.NET ...

    ASP.NET MVC4 Web编程中文版超清晰 PDF

    ASP.NET MVC4是一种基于微软.NET Framework的开源Web应用程序框架,专为构建可维护性和测试性强的Web应用而设计。这个框架结合了Model-View-Controller(MVC)设计模式、依赖注入、单元测试等现代开发理念,为开发者...

    【ASP.NET编程知识】ASP.NET MVC的四种验证编程方式.docx

    "ASP.NET MVC的四种验证编程方式" 在 ASP.NET MVC 中,验证是非常重要的一个步骤,它可以确保用户输入的数据合法和正确。今天,我们将讨论 ASP.NET MVC 中的四种验证编程方式。 手工验证绑定的参数 手工验证绑定的...

    ASP.NET MVC开发的课件2

    ### ASP.NET MVC 开发知识点详解 #### 一、ASP.NET MVC 概述及三层架构 - **ASP.NET MVC** 是一种流行的Web应用框架,由微软公司开发,它遵循了Model-View-Controller(模型-视图-控制器)设计模式。这种模式有助...

    ASP.NET MVC+EasyUI 框架+通用权限管理系统 + 源码

    EasyUI使前端界面的开发变得更加简单和高效,与ASP.NET MVC结合使用,可以快速构建出美观且交互性强的管理界面。 通用权限管理系统是该项目的关键部分,它的设计目标是实现对系统资源的细粒度访问控制。系统通常...

    AttributeRouting, 在 ASP.NET MVC和 Web API 中,使用操作的属性定义路由.zip

    AttributeRouting, 在 ASP.NET MVC和 Web API 中,使用操作的属性定义路由 AttributeRouting正在启动阅读了文档。如果你在使用这个库时遇到任何问题,请将它们记录在问题跟踪程序中。 注意,自微软将Attribute...

    ASP.NET MVC3快速入门

    ### ASP.NET MVC3快速入门知识点解析 #### 一、ASP.NET MVC3简介 **ASP.NET MVC3** 是一种用于构建动态网站的框架,它采用了**Model-View-Controller (MVC)** 架构模式,旨在简化开发过程并提高代码质量。这种模式...

    .NET MVC2介绍性文档

    - .NET 3.5的新特性,如扩展方法、匿名类型、对象初始化器和lambda表达式。 - LINQ技术,如Linq To Entity和Nhibernate作为ORM工具。 - 前端AJAX技术,如jQuery。 **编程思想**: - OOP(面向对象编程)。 - AOP...

    ExtJs+ASP.net+MVC

    5. **路由与HTTP请求处理**:ASP.NET MVC中的路由系统允许开发者自定义URL结构,以符合SEO优化和用户体验需求。控制器类处理HTTP请求,通过动作方法(Action Method)返回视图或JSON数据。 6. **AJAX通信**:ExtJS ...

    【ASP.NET编程知识】Asp.net mvc 权限过滤和单点登录(禁止重复登录).docx

    在ASP.NET MVC中,权限控制可以使用AuthorizeAttribute来实现。AuthorizeAttribute是一个抽象类,提供了多种方式来实现权限控制,例如通过角色、用户、权限等。下面是一个使用AuthorizeAttribute来实现权限控制的...

Global site tag (gtag.js) - Google Analytics