本文轉載自:http://www.cnblogs.com/soundcode/p/4054001.html
前言
本文描述了 ASP.NET Web API 如何將 HTTP 請求路由到控制器。
如果你熟悉Asp.Net MVC,Web API的路由與Asp.Net MVC的路由是非常類似的。這主要的區別就是Web API使用的是HTTP方法,而不是URI路徑來選擇Action。你也可以在Web API中使用MVC風格的路由。本文不需要有任何Asp.Net MVC的基礎。
Routing Tables路由表
在Asp.Net Web API中,一個控制器就是一個處理HTTP請求的類,控制器的public 方法被叫做action方法或者簡單的Aciton。當Web API接收到一個請求的時候,它將這個請求路由到一個Action。
為了確定那個Action被呼叫,這個框架使用了一個路由表。Visual Studio中Web API的專案模板會建立一個預設路由:
config.Routes.MapHttpRoute( name: "DefaultApi", routeTemplate: "api/{controller}/{id}", defaults: new { id = RouteParameter.Optional } );
這個路由是在WebApiConfig.cs檔案中定義的,該檔案位於App_Start目錄。
關於WebApiConfig類的更多資訊參閱“配置ASP.NET Web API”(暫未實現)
如果你要自己託管(self-host )Web API,你必須直接在HttpSelfHostConfiguration物件上設定路由表。更多資訊參閱“自託管Web API"。(暫未實現)
路由表中的每一個條目都包含一個路由模板。這個Web API預設的路由模版是"api/{controller}/{id}"。在這個模版中,“api”是一個文字式路徑片段,而{controller}和{id}則是佔位符變數。
當Web API框架接收一個HTTP請求時,它會試圖根據路由表中的一個路由模板來匹配其URI。如果無路由匹配,客戶端會接收到一個404(未找到)錯誤。例如,以下URI與這個預設路由的匹配:
- /api/contacts
- /api/contacts/1
- /api/products/gizmo1
然而,以下URI不匹配,因為它缺少“api”片段:
- /contacts/1
在路由中使用“api”的原因是為了避免與ASP.NET MVC的路由衝突。通過這種方式,可以用“/contacts”進入一個MVC控制器,而“/api/contacts”進入一個Web API控制器。當然,如果你不喜歡這種約定,你也可以修改這個預設路由表。
一旦一個匹配的路由被發現,Web API便會選擇相應的Controller和Action。
1.為了找到Controller,Web API會把“控制器”加到{controller}變數的值。
2.為了找到Action,Web API會查詢HTTP方法,然後尋找一個名稱以HTTP方法名開頭的方法。例如,對於一個Get請求,Web API會查詢一個以“Get…”開頭的動作,如“GetContact”或“GetAllContacts”等。這種約定只應用於GET、POST、PUT和DELETE方法。通過在你的Controller上使用attributes,你可以啟用其他的HTTP方法。稍後我們就會看到一個例子。
3.路由模版中其他的佔位變數,例如{id},將被對映成Action的引數。
讓我們來看一個簡單的例子,假設你定義了以下控制器:
public class ProductsController : ApiController { public void GetAllProducts() { } public IEnumerable<Product> GetProductById(int id) { } public HttpResponseMessage DeleteProduct(int id){ } }
以下是一些可能的HTTP請求,以及要被呼叫的每個動作:
注意,URI中的{id}片段如果出現,會被對映成Action的id引數。在這個例子中,這個控制器定義了兩個GET方法,一個帶有id引數的和一個不帶有id引數的。
另外要注意,POST請求是失敗的,因為該控制器未定義“Post…”方法。
Routing Variations路由變化
上一節描述了ASP.NET Web API基本的路由機制。本小節描述一些變化。
HTTP方法
替代使用HTTP方法的命名約定,你可以明確的為一個Action指定HTTP方法,通過以HttpGet、HttpPost、HttpPut或者HttpDelete屬性來對Action方法進行修飾。
在下列示例中,FindProduct方法被對映到GET請求:
public class ProductsController : ApiController { [HttpGet] public Product FindProduct(id) {} }
允許一個Action對應多個HTTP方法,或者允許除了Get、Put、Post、Delete方法之外的HTTP方法,需要使用AcceptVerbs註解屬性,它以HTTP方法列表作為引數。
public class ProductsController : ApiController { [AcceptVerbs("GET", "HEAD")] public Product FindProduct(id) { } // WebDAV method [AcceptVerbs("MKCOL")] public void MakeCollection() { } }
第一個方法:指示該動作接收HTTP的GET和HEAD方法(這個HEAD沒測試過)
第二個方法:WebDAV方法(基於Web的分散式著作與版本控制的HTTP方法,是一個擴充套件的HTTP方法
MKCOL是隸屬於WebDAV的一個方法,它在URI指定的位置建立集合(WebDAV更沒見過)
通過Action名稱路由
在預設的路由模版中,這個Web API使用HTTP方法去選擇Action。然而,你也可以在URI中建立包含動作名的路由:
routes.MapHttpRoute( name: "ActionApi", routeTemplate: "api/{controller}/{action}/{id}", defaults: new { id = RouteParameter.Optional } );
在這個路由模板中,{action}引數命名了控制器上的動作方法。採用這種風格的路由,需要使用註解屬性來指明所允許的HTTP方法。例如,假設你的控制器已有如下方法:
public class ProductsController : ApiController { [HttpGet] public string Details(int id); }
在這種情況下,一個Get請求"api/Products/Details/1"將會對映到這個這個Details方法。這種風格的路由類似於Asp.Net MVC,而且可能與RPC式的API相接近。(RPC風格不太懂,還沒查資料)
你也可以通過使用ActionName註解屬性來覆蓋動作名。在以下例子中,有兩個動作對映到“api/products/thumbnail/id”。一個支援GET,而另一個支援POST:
public class ProductsController : ApiController { [HttpGet] [ActionName("Thumbnail")] public HttpResponseMessage GetThumbnailImage(int id); [HttpPost] [ActionName("Thumbnail")] public void AddThumbnailImage(int id); }
Non-Actions
為了防止一個方法被作為一個動作所請求,可以使用NonAction註解屬性。它對框架發出訊號:這個方法不是一個動作,,即使它可能與路由規則匹配。