ASP.NET Core MVC 从入门到精通之路由

随着技术的发展,ASP.NET Core MVC也推出了好长时间,经过不断的版本更新迭代,已经越来越完善,本系列文章主要讲解ASP.NET Core MVC开发B/S系统过程中所涉及到的相关内容,适用于初学者,在校毕业生,或其他想从事ASP.NET Core MVC 系统开发的人员。 经过前几篇文章的讲解,初步了解ASP.NET Core MVC项目创建,启动运行,以及ASP.NET Core MVC的命名约定,创建控制器,视图,模型,接收参数,传递数据等内容,今天继续讲解ASP.NET Core MVC 路由等相关内容,仅供学习分享使用。

ASP.NET Core MVC 从入门到精通之路由

 

 

什么是路由?

 

路由是一种机制,主要是用于检查每一个用户请求,将用户请求映射到Action中,这一动作通过路由中间件来实现。ASP.NET Core MVC使用路由中间件来匹配传入请求的URL并将它们映射到操作(Action方法)。

 

默认路由

 

在通过模板创建ASP.NET Core MVC中,默认会添加路由中间件,并提供一种默认的路由映射规则和约束。

MapControllerRoute 用于创建单个路由。 单个路由命名为 default 路由。 大多数具有控制器和视图的应用都使用类似 default 路由的路由模板。如下所示:

 1 using Microsoft.AspNetCore.Server.Kestrel.Core;
 2 using System.Text.Encodings.Web;
 3 using System.Text.Unicode;
 4 
 5 var builder = WebApplication.CreateBuilder(args);
 6 
 7 // Add services to the container.
 8 builder.Services.AddControllersWithViews().AddJsonOptions(options =>
 9 {
10     options.JsonSerializerOptions.Encoder = JavaScriptEncoder.Create(UnicodeRanges.All);
11 });
12 
13 builder.Services.Configure<KestrelServerOptions>(options =>
14 {
15     options.AllowSynchronousIO = true;
16 });
17 
18 var app = builder.Build();
19 
20 // Configure the HTTP request pipeline.
21 if (!app.Environment.IsDevelopment())
22 {
23     app.UseExceptionHandler("/Home/Error");
24     // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
25     app.UseHsts();
26 }
27 
28 app.UseHttpsRedirection();
29 app.UseStaticFiles();
30 //1. 添加路由中间件EndpointRoutingMiddleware
31 app.UseRouting();
32 
33 app.UseAuthorization();
34 
35 //2.为控制器和Action添加一种路由映射规则,包括名称,规则,约束等
36 app.MapControllerRoute(
37     name: "default",
38     pattern: "{controller=Home}/{action=Index}/{id?}");
39 
40 app.Run();

在上述代码中,创建默认路由的关键代码如下所示:

1 app.MapControllerRoute(
2     name: "default",
3     pattern: "{controller=Home}/{action=Index}/{id?}");

其中各个属性说明:

  • name表示路由名称,默认值为default。
  • pattern 为匹配模板,默认值为"{controller=Home}/{action=Index}/{id?}"。其中默认controller=Home,表示缺省值为HomeController,action=Index,表示默认缺省值为Index。id后面的?表示为可空类型,可以根据需要来填写。

 

默认路由示例

 

  • 如:/ 表示controller和action都采用默认值,相当于/Home/Index。
  • 如:/Hello表示只指定了controller,action为默认缺省值,相当于/Hello/Index。
  • 如:/Hello/Test/5 根据路由匹配规则,提取出controller=HelloController , action=Test , id =5 。如果刚好存在此controller和action,则匹配导航成功。

创建默认路由的简便写法,可通过以下代码替代上面的app.MapControllerRoute,如下所示:

1 //创建默认传统路由的简便写法
2 app.MapDefaultControllerRoute();

以上是传统路由的示例。 之所以称为传统路由,是因为它为 URL 路径建立了一个约定:

  • 第一个路径段 {controller=Home} 映射到控制器名称。
  • 第二段 {action=Index} 映射到操作名称。
  • 第三段 {id?} 用于可选 id。 {id?} 中的 ? 使其成为可选。 id 用于映射到模型实体。

此传统路由映射具有以下特点:

  • 仅基于控制器和操作名称。
  • 不基于命名空间、源文件位置或方法参数。

使用默认路由进行传统路由可以创建应用,而无需为每个操作提出新的 URL 模式,有助于简化代码。使 UI 更具可预测性。

 

多个路由

 

在实际应用中,可以设置多个路由,为某些特定的需求设置专有路由。设置多个路由,如下所示:

1 //2.为控制器和Action添加一种路由映射规则,包括名称,规则,约束等
2 app.MapControllerRoute(
3     name: "blog",
4     pattern: "blog/{*article}",
5     defaults: new { controller = "Blog", action = "Article" });
6 
7 app.MapControllerRoute(
8     name: "default",
9     pattern: "{controller=Home}/{action=Index}/{id?}");

在上述代码中,名称为blog的路由,虽然采用的是传统路由,但只用于特定操作。

在创建多个路由时,以下几点需要注意:

  • blog 路由比 default 路由具有更高的匹配优先级,因为它最先添加。
  • 路由名称为路由指定逻辑名称,使用命名路由可以简化 URL 创建,在应用程序范围内必须唯一。
  • 默认情况下,传统路由依赖于顺序。 一般情况下,具有区域的路由应放在路由表中靠前的位置,因为它们比没有区域的路由更具体。也就是:特殊的在前面,通用的在后面

 

不明确操作

 

如果同一个请求有两个action都满足路由终结点的匹配,那么路由会进行如下处理:

  • 选择最佳的候选项。
  • 引发异常。

如下代码,在请求时,会两个都满足路由匹配规则:

 1 /// <summary>
 2 /// 请求需要编辑的学生信息
 3 /// </summary>
 4 /// <param name="id"></param>
 5 /// <returns></returns>
 6 public IActionResult Edit(int id)
 7 {
 8     var student = new Student()
 9     {
10         Id = 1,
11         Name = "公子小六",
12         Age = 21,
13         Sex = ""
14     };
15     return View();
16 }
17 
18 /// <summary>
19 /// 编辑后保存学生信息
20 /// </summary>
21 /// <param name="id"></param>
22 /// <param name="student"></param>
23 /// <returns></returns>
24 public IActionResult Edit(int id, Student student)
25 {
26     //保存学生
27     return View();
28 }

那么路由中间件就会跑出异常,如下所示:

ASP.NET Core MVC 从入门到精通之路由

 

 在这种情况下,要解析正确的路由,需要通过Http谓词来区分,只有当请求为Post时,才会请求Edit(int id,Student student);其他请求时【一般为Get】,匹配Edit(int id)。添加Http谓词后的代码如下:

 1 /// <summary>
 2 /// 请求需要编辑的学生信息
 3 /// </summary>
 4 /// <param name="id"></param>
 5 /// <returns></returns>
 6 public IActionResult Edit(int id)
 7 {
 8     var student = new Student()
 9     {
10         Id = 1,
11         Name = "公子小六",
12         Age = 21,
13         Sex = ""
14     };
15     return View();
16 }
17 
18 /// <summary>
19 /// 编辑后保存学生信息
20 /// </summary>
21 /// <param name="id"></param>
22 /// <param name="student"></param>
23 /// <returns></returns>
24 [HttpPost]
25 public IActionResult Edit(int id, Student student)
26 {
27     //保存学生
28     return View();
29 }

 

属性路由

 

属性路由一般用于WebAPI,使用一组属性将操作直接映射到路由模板,将应用功能建模为一组资源。属性路由使用[Route(template)]标记于controller或action中,示例如下所示:

 1 public class TestController : Controller
 2 {
 3 
 4     [Route("Test1")]
 5     [Route("Test1/Index")]
 6     [Route("Test1/Index/{id?}")]
 7     public IActionResult Index(int id)
 8     {
 9         ViewBag.Id = id;
10         return View();
11     }
12 
13     public IActionResult Test()
14     {
15         return View();
16     }
17 }

运行程序,在浏览器中,输入网址【https://localhost:7116/Test1/Index/10】,如下所示:

ASP.NET Core MVC 从入门到精通之路由

 

注意:属性路由中,也可用标记:[Route("[controller]/[action]")],效果可传统路由一致。

属性路由与传统路由对比

经过以上示例,属性路由与传统路由,主要由以下几点差异:

  •  属性路由需要更多输入才能指定路由,属性路由自定义比较强,更能精准控制路由。
  • 传统默认路由会更简洁地处理路由。 
  • 属性路由优先级高于传统路由。对于属性路由,控制器和操作名称在操作匹配中不起作用,除非使用标记替换。
  • 属性路由支持定义多个访问同一操作的路由,意味着每个路由属性都与操作方法上的每个路由属性相结合。
  • 属性路由支持使用与传统路由相同的内联语法,来指定可选参数、默认值和约束。如:[HttpPost("product14/{id:int}")]

 

保留关键字

 

在ASP.NET Core MVC项目中,会有一些关键字,作为路由参数名称,如下所示:

  • action
  • area
  • controller
  • handler
  • page

在 Razor 视图或 Razor 页面的上下文中保留以下关键字:

  • page
  • using
  • namespace
  • inject
  • section
  • inherits
  • model
  • addTagHelper
  • removeTagHelper

一个常见错误是使用 page 作为属性路由的路由参数。 这样做会导致与 URL 生成不一致和令人困惑的行为。错误示例如下所示:

1 public class TestController : Controller
2 {
3     [Route("/articles/{page}")]
4     public IActionResult ListArticles(int page)
5     {
6         return View(page);
7     }
8 }

 

Http谓词和路由模板

 

在ASP.NET Core MVC项目中,具有以下几种谓词,用于区分不同的请求方式:

  • [HttpGet]
  • [HttpPost]
  • [HttpPut]
  • [HttpDelete]
  • [HttpHead]
  • [HttpPatch]

ASP.NET Core 具有以下路由模板:

  • 所有 HTTP 谓词模板都是路由模板。
  • [Route]

 

混合路由:属性路由与传统路由

 

ASP.NET Core 应用可以混合使用传统路由和属性路由。 通常,对为浏览器提供 HTML 页的控制器使用传统路由,对为 API 提供服务 REST 的控制器使用属性路由。

操作既支持传统路由,也支持属性路由。 通过在控制器或操作上放置路由可实现属性路由。 不能通过传统路由访问定义属性路由的操作,反之亦然。 控制器上的任何路由属性都会使控制器中的所有操作使用属性路由。

属性路由和传统路由使用相同的路由引擎。

以上就是ASP.NET Core MVC从入门到精通之路由的全部内容,旨在抛砖引玉,一起学习,共同进步。

原文链接:https://www.cnblogs.com/hsiang/p/17308420.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:ASP.NET Core MVC 从入门到精通之路由 - Python技术站

(0)
上一篇 2023年4月19日
下一篇 2023年4月19日

相关文章

  • WPF+ASP.NET SignalR实现动态折线图的绘制

    下面是详细的攻略: 简介 本文介绍如何使用 WPF 和 ASP.NET SignalR 实现动态折线图的绘制。WPF 是一个用于创建 Windows 应用程序的 UI 框架,而 ASP.NET SignalR 是一个用于实现实时应用程序的框架,两者结合可以实现实时折线图的绘制。 准备工作 在开始实现动态折线图之前,我们需要准备以下工具: Visual Stu…

    C# 2023年6月3日
    00
  • net core下链路追踪skywalking安装和简单使用教程

    .NET Core下链路追踪SkyWalking安装和简单使用教程 在本攻略中,我们将详细讲解如何在.NET Core应用程序中安装和使用SkyWalking进行链路追踪,并提供两个示例说明。 SkyWalking简介 SkyWalking是一个开源的分布式系统追踪解决方案,可以帮助开发人员更好地了解应用程序的性能和行为。SkyWalking支持多种语言和框…

    C# 2023年5月16日
    00
  • C#实现AI五子棋游戏的示例代码

    作为网站的作者,我非常乐意为您提供关于“C#实现AI五子棋游戏的示例代码”的详细攻略。 首先,要实现AI五子棋游戏,我们需要使用C#语言,并结合人工智能算法。在这个攻略中,我会使用“极大极小值算法”来实现AI的五子棋游戏。 一、五子棋游戏的棋盘设计 我们可以使用一个二维数组来表示五子棋的棋盘,其中0表示空格,1表示黑子,-1表示白子。 示例代码如下: pri…

    C# 2023年6月7日
    00
  • 在 asp.net core 的中间件中返回具体的页面的实现方法

    在 ASP.NET Core 中,中间件是处理 HTTP 请求和响应的组件,可以是任何处理这些请求和响应的代码。在中间件中返回具体的页面,通常需要借助于 ASP.NET Core MVC 中的视图引擎和控制器。下面是具体的实现方法。 步骤1:添加 MVC 中间件服务 首先,需要在 ASP.NET Core 应用程序中添加 MVC 中间件服务。在 Startu…

    C# 2023年5月31日
    00
  • 浅谈C#中正则表达式的使用

    浅谈C#中正则表达式的使用 正则表达式是对字符模式的描述和匹配的一种语法工具,在C#中对正则表达式的支持非常好。本文将详细介绍在C#中如何使用正则表达式,包括正则表达式的语法、使用方法,以及常见的示例。 正则表达式的语法 正则表达式使用一些特殊字符表示模式的匹配规则,其中一些字符具有预定义的含义,也有一些字符需要使用转义等特殊处理。以下是正则表达式常用的字符…

    C# 2023年6月3日
    00
  • C#面向对象设计原则之里氏替换原则

    C#面向对象设计原则之里氏替换原则 什么是里氏替换原则 里氏替换原则(Liskov Substitution Principle,LSP)是面向对象设计中的一个基本原则。它重新定义了关于继承的条款。原始的定义是由“Barbara Liskov”于1987年提出的:“如果对于每一个类型为 T1 的对象 o1 都有类型为 T2 的对象 o2,使得以 T1 定义的…

    C# 2023年5月14日
    00
  • C# 如何调用C++ dll string类型返回

    C# 调用 C++ DLL 的过程中,若遇到需要返回 string 类型的情况,可以使用字符缓冲区来传递字符串,并通过指针参数来返回。 以下为详细步骤: 定义 C++ 端的 DLL 接口函数 在 C++ 中,需要定义一个导出函数用于将 C# 中的字符串传递到 DLL 中,例如以下代码段: // Example.cpp extern "C"…

    C# 2023年6月6日
    00
  • ASP.NET Core基础之请求处理管道

    ASP.NET Core基础之请求处理管道 在 ASP.NET Core 中,请求处理管道是一个非常重要的组件,它负责处理 HTTP 请求并生成 HTTP 响应。本攻略将详细介绍 ASP.NET Core 的请求处理管道。 请求处理管道的作用 ASP.NET Core 的请求处理管道负责处理 HTTP 请求并生成 HTTP 响应。请求处理管道可以帮助我们: …

    C# 2023年5月16日
    00
合作推广
合作推广
分享本页
返回顶部