[譯]ASP.NET Core 2.0 路由引擎詳解
本文介紹了ASP.NET Core 2.0 路由引擎詳解,分享給大家,具體如下:
問題
ASP.NET Core 2.0的路由引擎是如何工作的?
答案
創(chuàng)建一個(gè)空項(xiàng)目,為Startup類添加MVC服務(wù)和請求中間件:
public void ConfigureServices(IServiceCollection services)
{
services.AddMvc();
}
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
app.UseMvc(routes =>
{
routes.MapRoute(
name: "goto_one",
template: "one",
defaults: new { controller = "Home", action = "PageOne" });
routes.MapRoute(
name: "goto_two",
template: "two/{id?}",
defaults: new { controller = "Home", action = "PageTwo" });
routes.MapRoute(
name: "default",
template: "{controller=Home}/{action=Index}/{id?}");
});
}
創(chuàng)建一個(gè)控制器HomeController,來演示常規(guī)路由:
public class HomeController : Controller
{
public IActionResult Index()
{
return Content("Home/Index");
}
public IActionResult PageOne()
{
return Content("Home/One");
}
[HttpGet]
public IActionResult PageTwo()
{
return Content("(GET) Home/Two");
}
[HttpPost]
public IActionResult PageTwo(int id)
{
return Content($"(POST) Home/Two: {id}");
}
}
創(chuàng)建一個(gè)控制器WorkController,來演示特性路由:
[Route("work")]
public class WorkController : Controller
{
public IActionResult Index()
{
return Content("Work/Index");
}
[Route("one")]
public IActionResult PageOne()
{
return Content("Work/One");
}
[HttpGet("two")]
public IActionResult PageTwo()
{
return Content("(GET) Work/Two");
}
[HttpPost("two/{id?}")]
public IActionResult PageTwo(int id)
{
return Content($"(POST) Work/Two: {id}");
}
}
討論
ASP.NET Core的路由引擎可以將傳入的請求映射到控制器和它們的方法中。這是通過向請求管道中添加路由中間件實(shí)現(xiàn)的,具體來說是使用IRouteBuilder將URL規(guī)則(模板)映射到一個(gè)控制器的方法。
路由模板
路由模板可以使用字面值和標(biāo)記(標(biāo)識(shí)路由參數(shù))。在匹配一個(gè)路由時(shí),字面值會(huì)嚴(yán)格匹配URL中的文本,而標(biāo)記會(huì)被替換掉。
為了匹配一個(gè)模板,模板中必須包含控制器和方法標(biāo)記以便定位控制器方法(這是MVC的核心信息)。模板中的其它標(biāo)記被映射為方法的參數(shù)(通過模型綁定實(shí)現(xiàn))。
當(dāng)添加一個(gè)路由映射時(shí),可以為標(biāo)記提供缺省值。當(dāng)模板中不包含控制器和方法標(biāo)記時(shí)會(huì)很有用。模板也可以包含對(duì)應(yīng)于方法參數(shù)的可選標(biāo)記。
讓我們來看一個(gè)示例模板:
contact/{controller=Home}/{action=Index}/{id?}
注意如下幾點(diǎn):
1.標(biāo)記包含中大括號(hào)中。這里有三個(gè)標(biāo)記,分別是controller,action和id。
2.模板中包含一個(gè)字面值contact,它會(huì)匹配URL中的文本。
3.已經(jīng)為controller(Home)和action(Index)提供了默認(rèn)值。
4.可選標(biāo)記通過問號(hào)來聲明。
下面的URL會(huì)匹配這個(gè)模板:
- /contact/Home/Index/1: 所有標(biāo)記都有值。
- /contact/Home/Index: 忽略了可選標(biāo)記。
- /contact/Home: 忽略了action標(biāo)記,將使用默認(rèn)值Index。
- /contact: 忽略了controller和action標(biāo)記,將分別使用其默認(rèn)值Home和Index。
常規(guī)路由
常規(guī)路由為URL路徑建立一個(gè)約定, 例如給定一個(gè)模板:
1.第一個(gè)標(biāo)記映射到控制器
2.第二個(gè)標(biāo)記映射到方法
3.第三個(gè)標(biāo)記映射到可選的方法參數(shù)id
你也可以從模板中省略控制器和方法,只要你為它們提供缺省值就行了。比如下面的路由會(huì)映射到地址/one,因?yàn)橥ㄟ^defaults提供了所需的控制器和方法標(biāo)記:
routes.MapRoute(
name: "goto_one",
template: "one",
defaults: new { controller = "Home", action = "PageOne" });
注:請將此特定路由添加到通用路由之前,因?yàn)槁酚墒前凑斩x的順序執(zhí)行的,一旦某個(gè)路由匹配成功,則整個(gè)匹配流程就會(huì)終結(jié)。
由于路由中間件只使用了控制器和方法標(biāo)記來映射到一個(gè)控制器方法,因此同一個(gè)控制器中放置多個(gè)同名的的方法將會(huì)拋出異常。為了解決這個(gè)問題,可以使用方法上的IActionConstraint特性(比如HttpGet,HttpPost等特性):
[HttpGet("two")]
public IActionResult PageTwo()
{
return Content("(GET) Work/Two");
}
[HttpPost("two/{id?}")]
public IActionResult PageTwo(int id)
{
return Content($"(POST) Work/Two: {id}");
}
====start by sanshi=========================
為了觀察控制器中同名方法出現(xiàn)的異常,我們首先需要修改Configure()方法,添加開發(fā)時(shí)異常處理中間件:
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseMvc(routes => ....);
}
修改HomeController:
public IActionResult PageTwo()
{
return Content("(GET) Home/Two");
}
public IActionResult PageTwo(int id)
{
return Content($"(POST) Home/Two: {id}");
}
看似很正常的重載函數(shù),但是放到控制器中會(huì)拋出異常。
在瀏覽器地址欄敲入:http://localhost:65415/Home/PageTwo,觀看到異常頁面:
====end by sanshi=========================
特性路由
特性路由通過直接為控制器和方法提供路由模板來實(shí)現(xiàn)。
我們可以使用[Route]或者[HttpGet](或者其他動(dòng)詞)特性來指定模板。這些模板可以包含字面值和標(biāo)記(不能包含控制器和方法標(biāo)記)。
運(yùn)行時(shí),控制器的特性模板和方法的特性模板會(huì)被合并到一起,比如,在WorkController中,PageOne方法可以通過/work/one訪問:
[Route("work")]
public class WorkController : Controller
{
[Route("one")]
public IActionResult PageOne()
{
return Content("Work/One");
}
}
原文:https://tahirnaushad.com/2017/08/20/asp-net-core-mvc-routing/
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
.Net Core 中選項(xiàng)Options的具體實(shí)現(xiàn)
這篇文章主要介紹了.Net Core 中選項(xiàng)Options的具體實(shí)現(xiàn),文中運(yùn)用大量代碼對(duì)相關(guān)知識(shí)詳細(xì)介紹,感興趣的小伙伴可以參考一下這篇文章,希望對(duì)你有所幫助2021-09-09
asp.net下比較兩個(gè)等長字符串是否含有完全相同字符(忽略字符順序)
項(xiàng)目中遇到一個(gè)好玩的問題,需要比較兩個(gè)選擇區(qū)域選擇的文字是否一樣,就想到將這兩個(gè)區(qū)域中選中的文字鏈接起來進(jìn)行兩個(gè)字符串之間的比較2010-06-06
ASP.NET數(shù)據(jù)庫緩存依賴實(shí)例分析
這篇文章主要介紹了ASP.NET數(shù)據(jù)庫緩存依賴,以實(shí)例的形式分析總結(jié)了數(shù)據(jù)庫緩存依賴的原理與用法,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10
如何對(duì)ASP.NET網(wǎng)站實(shí)現(xiàn)靜態(tài)化
對(duì)于訪問量比較大的網(wǎng)站,網(wǎng)頁靜態(tài)化是一個(gè)比較可靠的解決方案。靜態(tài)化將顯著降低服務(wù)器的壓力,提升服務(wù)器處理能力。下面將介紹兩種不同的實(shí)現(xiàn)方法,并進(jìn)行對(duì)比。2015-09-09
URL重寫及干掉ASP.NET試圖狀態(tài)的實(shí)現(xiàn)方法
URL重寫已經(jīng)很普遍了,但基本上大部分的URL重寫都不支持頁面的相對(duì)路徑,所有如果想在已經(jīng)開發(fā)好的項(xiàng)目中添加還是有壓力的,第二就是例如微軟的那個(gè)URL重寫是根據(jù)正則表達(dá)式來處理的,那樣是很好,但也有不足之處,就是不方便定位到某個(gè)頁面只能有哪些參數(shù)2011-11-11
ASP.NET Core擴(kuò)展庫的相關(guān)功能介紹
這篇文章主要介紹了ASP.NET Core擴(kuò)展庫的相關(guān)功能,幫助大家更好的理解和學(xué)習(xí)使用.Net技術(shù),感興趣的朋友可以了解下2021-03-03
ASP.NET中使用GridView實(shí)現(xiàn)分級(jí)顯示的代碼
在實(shí)際項(xiàng)目開發(fā)中,往往需要用到在頁面上對(duì)列表的項(xiàng)目實(shí)現(xiàn)分級(jí)顯示,在 ASP.NET中沒有現(xiàn)成的控件。2010-06-06
ASP.NET.4.5.1+MVC5.0設(shè)置系統(tǒng)角色與權(quán)限(二)
這篇文章主要介紹了使用ASP.NET.4.5.1+MVC5.0構(gòu)建項(xiàng)目中設(shè)置系統(tǒng)角色的全部過程,十分的詳細(xì),附上全部源碼,推薦給想學(xué)習(xí).net+mvc的小伙伴們2015-01-01

