ASP.NET Core通过路由(Routing)设定,将定义的URL规则找到相对应行为;当使用者Request的URL满足特定规则条件时,则自动对应到相符合的行为处理。从ASP.NET就已经存在的架构,而且用法也很相似,只有些许的不同。
本篇将介绍ASP.NET Core的Router Middleware用法。

简单路由

之前 ASP.NET Core 2 学习笔记(三)有介绍到,可以透过Map处理一些简单路由,例如:

Startup.cs

// ...
public class Startup
{
// ...
public void Configure(IApplicationBuilder app)
{
// ...
app.Map("/first", mapApp =>
{
mapApp.Run(async context =>
{
await context.Response.WriteAsync("First. \r\n");
});
});
app.Map("/second", mapApp =>
{
mapApp.Run(async context =>
{
await context.Response.WriteAsync("Second. \r\n");
});
});
}
}

但要搭配ASP.NET Core MVC的话,简单路由就没这么好用了。
Router Middleware除了方便搭配ASP.NET Core MVC外,也可以比较弹性的使用路由定义。

路由注册

RouterMiddleware 的路由注册方式大致分为两种:

  • 全局注册。如:MapRoute
  • 局部注册。如:RouteAttribute

预设路由的顺序如下:

路由的Middleware需要Microsoft.AspNetCore.Routing套件。

Startup.csConfigureServices加入Routing的服务,并在Configure定义路由规则:

Startup.cs

// ...
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services.AddRouting();
} public void Configure(IApplicationBuilder app)
{
var defaultRouteHandler = new RouteHandler(context =>
{
var routeValues = context.GetRouteData().Values;
return context.Response.WriteAsync($"Route values: {string.Join(", ", routeValues)}");
}); var routeBuilder = new RouteBuilder(app, defaultRouteHandler);
routeBuilder.MapRoute("default", "{first:regex(^(default|home)$)}/{second?}"); routeBuilder.MapGet("user/{name}", context => {
var name = context.GetRouteValue("name");
return context.Response.WriteAsync($"Get user. name: {name}");
}); routeBuilder.MapPost("user/{name}", context => {
var name = context.GetRouteValue("name");
return context.Response.WriteAsync($"Create user. name: {name}");
}); var routes = routeBuilder.Build();
app.UseRouter(routes);
}
}

可以看到上面代码,建立了两个对象:

  • RouteHandler:这个对象如同简单路由的Run事件,当路由成立的时候,就会执行这个事件。
  • RouteBuilder:在这个对象定义路由规则,当Requset URL符合规则就会执行该事件。
    • MapRoute:预设的路由规则,可以支持正则表示式(Regular Expressions)。
    • MapGet & MapPost
      HTTP Method路由:同样的URL可以通过不同的HTTP Method,对应不同的事件。

第一个路由MapRoute定义:

  • URL第一层通过正则表示式必需是default或home,并放到路由变量first中。
  • URL第二层可有可无,如果有的话,放到路由变量second中。

第二个路由MapGet定义:

  • 指定要是HTTP Get
  • URL第一层必需是user。
  • URL第二层必需要有值,放到路由变量name中。

第三个路由MapPost定义:

  • 指定要是HTTP Post
  • URL第一层必需是user。
  • URL第二层必需要有值,放到路由变量name中。

以上设定的路由结果如下:

  • http://localhost:5000/default会显示:
    Route values: [first, default]
  • http://localhost:5000/home/about会显示:
    Route values: [first, home], [second, about]
  • http://localhost:5000/user/snaildev通过HTTP Get会显示:
    Get user. name: snaildev
  • http://localhost:5000/user/snaildev通过HTTP Post会显示:
    Create user. name: snaildev

MVC 路由

MVC路由使用跟上面范例差不多,只是把事件指向ControllerAction
ASP.NET Core MVC注册路由规则的方式跟ASP.NET MVC差不多。
可以注册多个MapRoute,每个Request会经过这些RouterMiddleware找到对应Action。
先被执行到的路由,后面就会被跳过,所以越通用的越写在下面,如下:

Startup.cs

public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services.AddMvc();
}
public void Configure(IApplicationBuilder app)
{
app.UseMvc(routes =>
{
routes.MapRoute(
name: "about",
template: "about",
defaults: new { controller = "Home", action = "About" }
);
routes.MapRoute(
name: "default",
template: "{controller=Home}/{action=Index}/{id?}"
);
// 跟上面设置的 default 效果一样
//routes.MapRoute(
// name: "default",
// template: "{controller}/{action}/{id?}",
// defaults: new { controller = "Home", action = "Index" }
//);
});
}
}

以上设定的路由结果如下:

  • http://localhost:5000 会对应到HomeController 的Index()。
  • http://localhost:5000/about 会对应到HomeController 的About()。
  • http://localhost:5000/home/test 会对应到HomeController 的Test()。

RouteAttribute

预设RouteAttribute的优先顺序高于Startup注册的MapRoute,所以当使用[Route]后,原本的MapRoute将不再对Controller或Action产生作用。

[Route("[controller]")]
public class UserController : Controller
{
[Route("")]
public IActionResult Profile()
{
return View();
} [Route("change-password")]
public IActionResult ChangePassword()
{
return View();
} [Route("[action]")]
public IActionResult Other()
{
return View();
}
}

以上设定的路由结果如下:

  • http://localhost:5000/user 会对应到UserController 的Profile()。
  • http://localhost:5000/user/change-password 会对应到UserController 的ChangePassword()。
  • http://localhost:5000/user/other 会对应到UserController 的Other()。

若Controller设定了[Route],Action就要跟着加[Route],不然会发生错误。

如果只有特定的Action 需要改路由,也可以只加Action。如下:

public class UserController : Controller
{
public IActionResult Profile()
{
return View();
} [Route("change-password")]
public IActionResult ChangePassword()
{
return View();
} public IActionResult Other()
{
return View();
}
}
  • http://localhost:5000/user/profile 会对应到UserController 的Profile()。
  • http://localhost:5000/change-password 会对应到UserController 的ChangePassword()。
  • http://localhost:5000/user/other 会对应到UserController 的Other()。

注意!如果[Route]是设定在Action,路径是由网站根路径开始算。

如果要用MVC预设路由,可以使用 ASP.NET Core 2 学习笔记(六)MVC中 提到的:

  • UseMvcWithDefaultRoute

这个是ASP.NET Core的预设路由,会将Request来的URL找到对应的Controller及Action。

参考

Routing in ASP.NET Core

老司机发车啦:https://github.com/SnailDev/SnailDev.NETCore2Learning

ASP.NET Core 2 学习笔记(七)路由的更多相关文章

  1. ASP.NET Core 2 学习笔记(十二)REST-Like API

    Restful几乎已算是API设计的标准,通过HTTP Method区分新增(Create).查询(Read).修改(Update)和删除(Delete),简称CRUD四种数据存取方式,简约又直接的风 ...

  2. sql server 关于表中只增标识问题 C# 实现自动化打开和关闭可执行文件(或 关闭停止与系统交互的可执行文件) ajaxfileupload插件上传图片功能,用MVC和aspx做后台各写了一个案例 将小写阿拉伯数字转换成大写的汉字, C# WinForm 中英文实现, 国际化实现的简单方法 ASP.NET Core 2 学习笔记(六)ASP.NET Core 2 学习笔记(三)

    sql server 关于表中只增标识问题   由于我们系统时间用的过长,数据量大,设计是采用自增ID 我们插入数据的时候把ID也写进去,我们可以采用 关闭和开启自增标识 没有关闭的时候 ,提示一下错 ...

  3. Asp.Net Core WebApi学习笔记(四)-- Middleware

    Asp.Net Core WebApi学习笔记(四)-- Middleware 本文记录了Asp.Net管道模型和Asp.Net Core的Middleware模型的对比,并在上一篇的基础上增加Mid ...

  4. ASP.NET Core 2 学习笔记(十三)Swagger

    Swagger也算是行之有年的API文件生成器,只要在API上使用C#的<summary />文件注解标签,就可以产生精美的线上文件,并且对RESTful API有良好的支持.不仅支持生成 ...

  5. ASP.NET Core 2 学习笔记(十)视图

    ASP.NET Core MVC中的Views是负责网页显示,将数据一并渲染至UI包含HTML.CSS等.并能痛过Razor语法在*.cshtml中写渲染画面的程序逻辑.本篇将介绍ASP.NET Co ...

  6. ASP.NET Core 2 学习笔记(一)开始

    原文:ASP.NET Core 2 学习笔记(一)开始 来势汹汹的.NET Core似乎要取代.NET Framework,ASP.NET也随之发布.NET Core版本.虽然名称沿用ASP.NET, ...

  7. 使用Visual Studio Code开发Asp.Net Core WebApi学习笔记(七)-- 结构化配置

    本篇将记录.Net Core里颇有特色的结构化配置的使用方法. 相比较之前通过Web.Config或者App.Config配置文件里使用xml节点定义配置内容的方式,.Net Core在配置系统上发生 ...

  8. 使用Visual Studio Code开发Asp.Net Core WebApi学习笔记(四)-- Middleware

    本文记录了Asp.Net管道模型和Asp.Net Core的Middleware模型的对比,并在上一篇的基础上增加Middleware功能支持. 在演示Middleware功能之前,先要了解一下Asp ...

  9. Asp .Net core 2 学习笔记(3) —— 静态文件

    这个系列的初衷是便于自己总结与回顾,把笔记本上面的东西转移到这里,态度不由得谨慎许多,下面是我参考的资源: ASP.NET Core 中文文档目录 官方文档 记在这里的东西我会不断的完善丰满,对于文章 ...

随机推荐

  1. Pie(浮点数二分)

    Pie http://poj.org/problem?id=3122 Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 2454 ...

  2. 未能加载文件或程序集“AjaxControlToolkit”或它的某一个依赖项

    对于这个问题,网上的解答都大同小异,最多的就是Bin文件夹下没有dll文件,引用路径问题.但我碰到的问题偏偏不是这个,而是没有一个人给出方法的问题.其实问题很简单,也很低级:IIS上发布网站的时候把整 ...

  3. IO模型与select,poll,epoll

    五种:阻塞,非阻塞,IO复印,信号驱动,异步. select,poll,epoll select: 典型用32个32位的整数表示1024个描述符,并发的局限. poll:功能同上,但数据结构不一样(链 ...

  4. PAT L1-009 N个数求和(模拟分数加法)

    本题的要求很简单,就是求N个数字的和.麻烦的是,这些数字是以有理数“分子/分母”的形式给出的,你输出的和也必须是有理数的形式. 输入格式: 输入第一行给出一个正整数N(<=100).随后一行按格 ...

  5. mysql基本的增删改查和条件语句

    增 insert into 表名(列名,列名......) values("test1",23),("test2",23),("test3" ...

  6. python之多并发socket

    先看socket多并发的服务端的代码,这里是用多线程实现的多并发socketserver import socketserver # socketserver有四个基本的类,后两个不常用,这4个类处理 ...

  7. 分割回文串 · Palindrome Partitioning

    [抄题]: 给定一个字符串s,将s分割成一些子串,使每个子串都是回文串. 返回s所有可能的回文串分割方案. 给出 s = "aab",返回 [ ["aa", & ...

  8. [leetcode]721. Accounts Merge账户合并

    Given a list accounts, each element accounts[i] is a list of strings, where the first element accoun ...

  9. TextView 小技巧

    1.TextView的单行显示 最后空间不够的画变为省略号显示 textView中让结尾变为省略号android:ellipsize = "end" 省略号在结尾tv.setEll ...

  10. handler------post传送方式

    package com.qianfeng.gp08_day26_hanlder2; import android.os.Bundle; import android.os.Handler; impor ...